Using configuration files

Usually, you may want to allow the user of the plugin to change the specifics of its behavior. Moreover, your plugin might need to have a permanent data store for some of its internal data.

Whatever the reason, BepInEx provides a built-in ConfigFile class for simple configuration files. The format is based on INI with some syntax from TOML, which allows you to save most of the basic data types.

Note

Using BepInEx's configuration API is optional. You can always provide a custom way to serialize and deserialize data for your plugin.

In this part, we will go through the core API for reading and writing configuration files.

Using configuration files in plugins

Inside the plugin, you get access to Config property that is a preconfigured configuration file.
The file is saved in BepInEx\config\<GUID>.cfg where <GUID> is the GUID of your plugin.

To access and create configuration values, you first need to define them with Bind<T>(String, String, T, String). Configuration initialization is often done in plugin startup code.

Example:

using BepInEx;
using BepInEx.Configuration;

namespace MyFirstPlugin
{
    [BepInPlugin(PluginInfo.PLUGIN_GUID, PluginInfo.PLUGIN_NAME, PluginInfo.PLUGIN_VERSION)]
    public class Plugin : BaseUnityPlugin
    {
        private ConfigEntry<string> configGreeting;
        private ConfigEntry<bool> configDisplayGreeting;

        private void Awake()
        {
            configGreeting = Config.Bind("General",      // The section under which the option is shown
                                         "GreetingText",  // The key of the configuration option in the configuration file
                                         "Hello, world!", // The default value
                                         "A greeting text to show when the game is launched"); // Description of the option to show in the config file

            configDisplayGreeting = Config.Bind("General.Toggles", 
                                                "DisplayGreeting",
                                                true,
                                                "Whether or not to show the greeting text");
            // Test code
            Logger.LogInfo("Hello, world!");
        }
    }
}
Tip

Instead of using the plugin startup method, you can also define wrappers inside the constructor. Moreover, you do not need to define all options at once and instead create them on demand!

After defining the values, you can use them right away with Value:

// Instead of just Debug.Log("Hello, world!")
if(configDisplayGreeting.Value)
    Logger.LogInfo(configGreeting.Value);

When you compile your plugin and run the game with it for the first time, the configuration file will be automatically generated.
In the case of this example, the following configuration file is created in BepInEx\config\MyFirstPlugin.cfg:

[General]

## A greeting text to show when the game is launched
# Setting type: String
# Default value: Hello, world!
GreetingTest = Hello, world!

[General.Toggles]

## Whether or not to show the greeting text
# Setting type: Boolean
# Default value: True
DisplayGreeting = true

Notice the similarities between the calls to Bind<T>(String, String, T, String) and the generated configuration file.

Creating configuration files manually

In some cases (e.g. preloader patchers, non-plugin DLLs), you may want to create a configuration file manually.

This can be done quickly by creating a new instance of ConfigFile:

// Create a new configuration file.
// First argument is the path to where the configuration is saved
// Second arguments specifes whether to create the file right away or whether to wait until any values are accessed/written
var customFile = new ConfigFile(Path.Combine(Paths.ConfigPath, "custom_config.cfg"), true);

// You can now create configuration wrappers for it
var userName = customFile.Bind("General",
    "UserName",
    "Deuce",
    "Name of the user");

// In plug-ins, you can still access the default configuration file
var configGreeting = Config.Bind("General", 
    "GreetingTest",
    "Hello, world!", 
    "A greeting text to show when the game is launched");
Note

Notice that we use Paths class to get the path to BepInEx\config. In general, it is recommended to use the paths provided in Paths instead of manually trying to locate the directories.

Summary

In this part, we briefly overviewed the use of configuration files.

Next, you should get better accustomed to the additional API provided in ConfigFile and ConfigEntry<T> if you want to use configuration files supplied by BepInEx.
The additional API allows you to manually save and reload configuration as well.

This part concludes the basic plugin tutorial. Feel free to refer to BepInEx API Docs for extensive information on all methods that BepInEx provides. Check through some of the advanced guides for information on how to use BepInEx: