# Database

Nitrite database is a serverless, embedded, and self-contained database for Flutter. It is an open-source project that provides a simple API for persistent data storage. Nitrite database is designed to be lightweight, fast, and easy to use.

# Creating a Database

Nitrite database can be created in-memory or on-disk. By default, Nitrite database is created in-memory. To create a database on-disk, you need to add a storage module dependency to your project. More details about storage modules can be found here.

To create a database, you need to use NitriteBuilder class. To get an instance of NitriteBuilder, you need to call builder() method on Nitrite class.

var builder = Nitrite.builder();

# In-memory Database

If you don't load any on-disk storage module, then Nitrite will create an in-memory database. The below code snippet shows how to create a database in-memory.

var db = await Nitrite.builder()
    .openOrCreate();

# On-disk Database

The below code snippet shows how to create a database on-disk.

# Hive Backed Database

var storeModule = HiveModule.withConfig()
    .crashRecovery(true)
    .path("/path/to/my.db")
    .build();

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .openOrCreate();

More details about Hive configuration can be found here.

# NitriteBuilder

NitriteBuilder is the builder class for Nitrite database. It provides methods to configure and create a Nitrite database instance.

# Open or Create a Database

To open or create a database, you need to call openOrCreate() method on NitriteBuilder instance. This method returns a Future<Nitrite> instance.

If no StoreModule is configured, then Nitrite will create an in-memory database. If a StoreModule is configured, then Nitrite will create an on-disk database. If the database file does not exist, then Nitrite will create a new database file. If the database file exists, then Nitrite will open the database file.

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .openOrCreate();

# Securing a Database

To secure a database, you need to call openOrCreate() method with username and password on NitriteBuilder instance. This method returns a Future<Nitrite> instance with the given username and password.

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .openOrCreate(username: "user", password: "password");

# Registering an EntityConverter

Nitrite database uses a mapper to map a Dart object to a document and vice versa. By default, Nitrite uses SimpleNitriteMapper as its mapper. This mapper uses EntityConverters to map a Dart object to a document and vice versa. To register an EntityConverter, you need to call registerEntityConverter() method on NitriteBuilder instance. This method returns the same NitriteBuilder instance.

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .registerEntityConverter(MyEntityConverter())
    .openOrCreate();

More details about EntityConverter can be found here.

# Loading a Module

Nitrite database is modular in nature. It provides various modules to extend its functionality. To load a module, you need to call loadModule() method on NitriteBuilder instance. This method returns the same NitriteBuilder instance.

# Loading a Storage Module

var db = Nitrite.builder()
    .loadModule(storeModule)
    .openOrCreate();

More on the Nitrite's module system can be found here.

# Adding Migration Steps

Nitrite database supports schema migration. To configure a migration step, you need to call addMigrations() method on NitriteBuilder instance. This method returns the same NitriteBuilder instance.

var migration = Migration(
    initialSchemaVersion,
    2,
    (instructionSet) {
        instructionSet.forDatabase().addUser('test-user', 'test-password');

        instructionSet
            .forRepository<OldClass>(key: 'demo1')
            .renameRepository('new')
            .changeDataType('empId', (value) => int.parse(value))
            .changeIdField(Fields.withNames(['uuid']), Fields.withNames(['empId']))
            .deleteField('uuid')
            .renameField('lastName', 'familyName')
            .addField('fullName', generator: (document) =>
                '${document['firstName']} ${document['familyName']}',
            )
            .dropIndex(['firstName'])
            .dropIndex(['literature.text'])
            .changeDataType('literature.ratings', (value) => (value as double).round());
    },
);

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .addMigrations([migration])
    .openOrCreate();

More on the schema migration can be found here.

# Current Schema Version

To get the current schema version of a database, you need to call schemaVersion() method on NitriteBuilder instance. This method returns the current schema version of the database.

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .schemaVersion(2);
    .openOrCreate();

# Field Separator Character

To configure the field separator character, you need to call fieldSeparator() method on NitriteBuilder instance. This method returns the same NitriteBuilder instance.

It is used to separate field names in a nested document. For example, if a document has a field address which is a nested document, then the field street of the nested document can be accessed using address.street syntax.

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .fieldSeparator('.')
    .openOrCreate();

# Disable Repository Type Validation

To disable repository type validation, you need to call disableRepositoryTypeValidation() method on NitriteBuilder instance. This method returns the same NitriteBuilder instance.

Repository type validation is a feature in Nitrite that ensures the type of the objects stored in the repository can be converted to and from the document. By default, the repository type validation is enabled. If you disable it, and if you try to store an object that cannot be converted to a document, Nitrite will throw an exception during the operation.

var db = await Nitrite.builder()
    .loadModule(storeModule)
    .disableRepositoryTypeValidation()
    .openOrCreate();