* Docs: edit of 3 files in backend contributor guide * Update contribute/backend/database.md Co-authored-by: brendamuir <100768211+brendamuir@users.noreply.github.com> * Fix from review * Update contribute/backend/database.md * Update contribute/backend/errors.md * Update contribute/backend/errors.md * Fix header levels * Style edits * Prettier fixes --------- Co-authored-by: brendamuir <100768211+brendamuir@users.noreply.github.com>
5.5 KiB
Database
Grafana uses databases to persist settings between restarts. If you don't specify one, Grafana creates a SQLite3 database file on your local disk. This guide explains how to store and retrieve data from the default or other databases.
Supported databases and services
Grafana supports the following databases:
Grafana uses the XORM framework for persisting objects to the database. For more information on how to use XORM, refer to the documentation.
Services don't use XORM directly. Instead, services use the SQL store, a special type of service that provides an abstraction for the database layer. There are two ways of using the sqlstore: using sqlstore handlers, and using the SQLStore instance.
sqlstore handlers
Deprecated: We are deprecating
sqlstorehandlers in favor of using theSQLStoreobject directly in each service. Since most services still use thesqlstorehandlers, we still want to explain how they work.
The sqlstore package allows you to register command handlers that either store or retrieve objects from the database. The sqlstore handlers are similar to services:
- Services are command handlers that contain business logic.
sqlstorehandlers are command handlers that access the database.
Register a sqlstore handler
Deprecated: Refer to the deprecation note for
sqlstorehandlers.
To register a handler:
- Create a new file,
myrepo.go, in thesqlstorepackage. - Create a command handler.
- Register the handler in the
initfunction:
func init() {
bus.AddHandlerCtx("sql", DeleteDashboard)
}
func DeleteDashboard(ctx context.Context, cmd *models.DeleteDashboardCommand) error {
return inTransactionCtx(ctx, func(sess *DBSession) error {
_, err := sess.Exec("DELETE FROM dashboards WHERE dashboard_id=?", cmd.DashboardID)
return err
})
}
Here, inTransactionCtx is a helper function in the sqlstore package that provides a session, that lets you execute SQL statements.
SQLStore
As opposed to a sqlstore handler, the SQLStore is a service itself. Like the handler, the SQLStore is responsible for storing and retrieving objects, to and from the database.
To use the SQLStore, inject it in your service struct:
type MyService struct {
SQLStore *sqlstore.SQLStore `inject:""`
}
You can now make SQL queries in any of your command handlers or event listeners:
func (s *MyService) DeleteDashboard(ctx context.Context, cmd *models.DeleteDashboardCommand) error {
if err := s.SQLStore.WithDbSession(ctx, func(sess *db.Session) error {
_, err := sess.Exec("DELETE FROM dashboards WHERE dashboard_id=?", cmd.DashboardID)
return err
})
}
For transactions, use the WithTransactionalDbSession method instead.
Migrations
As your use of Grafana evolves, you may need to create schema migrations for one or more database tables.
To see all the types of migrations you can add, refer to migrations.go.
Before you add a migration, make sure that you:
- Never change a migration that has been committed and pushed to
main. - Always add new migrations, to change or undo previous migrations.
Add a migration using one of the following methods:
- Add migrations in the
migrationspackage. - Implement the
DatabaseMigratorfor the service.
Important: If there are previous migrations for a service, use that method. Don't add migrations using both methods or you risk running migrations in the wrong order.
Add migrations in migrations package
Most services have their migrations located in the migrations package.
To add a migration:
-
Open the migrations.go file.
-
In the
AddMigrationsfunction, find theaddXxxMigrationfunction for the service you want to create a migration for. -
At the end of the
addXxxMigrationfunction, register your migration (refer to the following example).
Note: We no longer recommend putting migrations behind feature flags because this could cause the migration to skip integration testing.
Implement DatabaseMigrator
During initialization, SQL store queries the service registry, and runs migrations for every service that implements the DatabaseMigrator interface.
To add a migration:
- If needed, add the
AddMigration(mg *migrator.Migrator)method to the service. - At the end of the
AddMigrationmethod, register your migration:
func (s *MyService) AddMigration(mg *migrator.Migrator) {
// ...
mg.AddMigration("Add column age", NewAddColumnMigration(table, &Column{
Name: "age",
Type: migrator.DB_BigInt,
Nullable: true,
}))
}