The gcp:firebase/databaseInstance:DatabaseInstance resource, part of the Pulumi GCP provider, provisions Firebase Realtime Database instances: their type (default or user), region, and operational state. This guide focuses on three capabilities: user database creation, default database provisioning, and state management.
Database instances require a Firebase-enabled GCP project with the firebasedatabase.googleapis.com API activated. User databases require a Blaze (pay-as-you-go) plan. The examples are intentionally small. Combine them with your own security rules and application integration.
Create a user database in a specific region
Most deployments start by creating a user database in a chosen region for multi-region data isolation.
import * as pulumi from "@pulumi/pulumi";
import * as gcp from "@pulumi/gcp";
const basic = new gcp.firebase.DatabaseInstance("basic", {
project: "my-project-name",
region: "us-central1",
instanceId: "active-db",
});
import pulumi
import pulumi_gcp as gcp
basic = gcp.firebase.DatabaseInstance("basic",
project="my-project-name",
region="us-central1",
instance_id="active-db")
package main
import (
"github.com/pulumi/pulumi-gcp/sdk/v9/go/gcp/firebase"
"github.com/pulumi/pulumi/sdk/v3/go/pulumi"
)
func main() {
pulumi.Run(func(ctx *pulumi.Context) error {
_, err := firebase.NewDatabaseInstance(ctx, "basic", &firebase.DatabaseInstanceArgs{
Project: pulumi.String("my-project-name"),
Region: pulumi.String("us-central1"),
InstanceId: pulumi.String("active-db"),
})
if err != nil {
return err
}
return nil
})
}
using System.Collections.Generic;
using System.Linq;
using Pulumi;
using Gcp = Pulumi.Gcp;
return await Deployment.RunAsync(() =>
{
var basic = new Gcp.Firebase.DatabaseInstance("basic", new()
{
Project = "my-project-name",
Region = "us-central1",
InstanceId = "active-db",
});
});
package generated_program;
import com.pulumi.Context;
import com.pulumi.Pulumi;
import com.pulumi.core.Output;
import com.pulumi.gcp.firebase.DatabaseInstance;
import com.pulumi.gcp.firebase.DatabaseInstanceArgs;
import java.util.List;
import java.util.ArrayList;
import java.util.Map;
import java.io.File;
import java.nio.file.Files;
import java.nio.file.Paths;
public class App {
public static void main(String[] args) {
Pulumi.run(App::stack);
}
public static void stack(Context ctx) {
var basic = new DatabaseInstance("basic", DatabaseInstanceArgs.builder()
.project("my-project-name")
.region("us-central1")
.instanceId("active-db")
.build());
}
}
resources:
basic:
type: gcp:firebase:DatabaseInstance
properties:
project: my-project-name
region: us-central1
instanceId: active-db
The instanceId must be globally unique and cannot be reused after deletion. The region determines where data is stored; check available regions in the Firebase documentation. Without specifying type, the resource defaults to USER_DATABASE. The database starts in ACTIVE state unless you set desiredState to DISABLED.
Create a disabled user database
Teams often pre-provision databases in a disabled state to prevent accidental writes before configuration is complete.
import * as pulumi from "@pulumi/pulumi";
import * as gcp from "@pulumi/gcp";
const full = new gcp.firebase.DatabaseInstance("full", {
project: "my-project-name",
region: "europe-west1",
instanceId: "disabled-db",
type: "USER_DATABASE",
desiredState: "DISABLED",
});
import pulumi
import pulumi_gcp as gcp
full = gcp.firebase.DatabaseInstance("full",
project="my-project-name",
region="europe-west1",
instance_id="disabled-db",
type="USER_DATABASE",
desired_state="DISABLED")
package main
import (
"github.com/pulumi/pulumi-gcp/sdk/v9/go/gcp/firebase"
"github.com/pulumi/pulumi/sdk/v3/go/pulumi"
)
func main() {
pulumi.Run(func(ctx *pulumi.Context) error {
_, err := firebase.NewDatabaseInstance(ctx, "full", &firebase.DatabaseInstanceArgs{
Project: pulumi.String("my-project-name"),
Region: pulumi.String("europe-west1"),
InstanceId: pulumi.String("disabled-db"),
Type: pulumi.String("USER_DATABASE"),
DesiredState: pulumi.String("DISABLED"),
})
if err != nil {
return err
}
return nil
})
}
using System.Collections.Generic;
using System.Linq;
using Pulumi;
using Gcp = Pulumi.Gcp;
return await Deployment.RunAsync(() =>
{
var full = new Gcp.Firebase.DatabaseInstance("full", new()
{
Project = "my-project-name",
Region = "europe-west1",
InstanceId = "disabled-db",
Type = "USER_DATABASE",
DesiredState = "DISABLED",
});
});
package generated_program;
import com.pulumi.Context;
import com.pulumi.Pulumi;
import com.pulumi.core.Output;
import com.pulumi.gcp.firebase.DatabaseInstance;
import com.pulumi.gcp.firebase.DatabaseInstanceArgs;
import java.util.List;
import java.util.ArrayList;
import java.util.Map;
import java.io.File;
import java.nio.file.Files;
import java.nio.file.Paths;
public class App {
public static void main(String[] args) {
Pulumi.run(App::stack);
}
public static void stack(Context ctx) {
var full = new DatabaseInstance("full", DatabaseInstanceArgs.builder()
.project("my-project-name")
.region("europe-west1")
.instanceId("disabled-db")
.type("USER_DATABASE")
.desiredState("DISABLED")
.build());
}
}
resources:
full:
type: gcp:firebase:DatabaseInstance
properties:
project: my-project-name
region: europe-west1
instanceId: disabled-db
type: USER_DATABASE
desiredState: DISABLED
Setting type to USER_DATABASE explicitly documents intent, though it’s the default. The desiredState property controls whether the database accepts connections; DISABLED prevents all reads and writes until you change it to ACTIVE. This is useful for staging environments or databases awaiting security rule configuration.
Provision the project default database
Every Firebase project can have one default database that cannot be deleted once created. This example shows the full setup including API enablement.
import * as pulumi from "@pulumi/pulumi";
import * as gcp from "@pulumi/gcp";
import * as time from "@pulumiverse/time";
const _default = new gcp.organizations.Project("default", {
projectId: "rtdb-project",
name: "rtdb-project",
orgId: "123456789",
deletionPolicy: "DELETE",
labels: {
firebase: "enabled",
},
});
const firebase = new gcp.projects.Service("firebase", {
project: _default.projectId,
service: "firebase.googleapis.com",
});
const defaultProject = new gcp.firebase.Project("default", {project: _default.projectId}, {
dependsOn: [firebase],
});
const firebaseDatabase = new gcp.projects.Service("firebase_database", {
project: defaultProject.project,
service: "firebasedatabase.googleapis.com",
});
const wait60Seconds = new time.Sleep("wait_60_seconds", {createDuration: "60s"}, {
dependsOn: [firebaseDatabase],
});
const defaultDatabaseInstance = new gcp.firebase.DatabaseInstance("default", {
project: defaultProject.project,
region: "us-central1",
instanceId: "rtdb-project-default-rtdb",
type: "DEFAULT_DATABASE",
}, {
dependsOn: [wait60Seconds],
});
import pulumi
import pulumi_gcp as gcp
import pulumiverse_time as time
default = gcp.organizations.Project("default",
project_id="rtdb-project",
name="rtdb-project",
org_id="123456789",
deletion_policy="DELETE",
labels={
"firebase": "enabled",
})
firebase = gcp.projects.Service("firebase",
project=default.project_id,
service="firebase.googleapis.com")
default_project = gcp.firebase.Project("default", project=default.project_id,
opts = pulumi.ResourceOptions(depends_on=[firebase]))
firebase_database = gcp.projects.Service("firebase_database",
project=default_project.project,
service="firebasedatabase.googleapis.com")
wait60_seconds = time.Sleep("wait_60_seconds", create_duration="60s",
opts = pulumi.ResourceOptions(depends_on=[firebase_database]))
default_database_instance = gcp.firebase.DatabaseInstance("default",
project=default_project.project,
region="us-central1",
instance_id="rtdb-project-default-rtdb",
type="DEFAULT_DATABASE",
opts = pulumi.ResourceOptions(depends_on=[wait60_seconds]))
package main
import (
"github.com/pulumi/pulumi-gcp/sdk/v9/go/gcp/firebase"
"github.com/pulumi/pulumi-gcp/sdk/v9/go/gcp/organizations"
"github.com/pulumi/pulumi-gcp/sdk/v9/go/gcp/projects"
"github.com/pulumi/pulumi/sdk/v3/go/pulumi"
"github.com/pulumiverse/pulumi-time/sdk/go/time"
)
func main() {
pulumi.Run(func(ctx *pulumi.Context) error {
_default, err := organizations.NewProject(ctx, "default", &organizations.ProjectArgs{
ProjectId: pulumi.String("rtdb-project"),
Name: pulumi.String("rtdb-project"),
OrgId: pulumi.String("123456789"),
DeletionPolicy: pulumi.String("DELETE"),
Labels: pulumi.StringMap{
"firebase": pulumi.String("enabled"),
},
})
if err != nil {
return err
}
firebase, err := projects.NewService(ctx, "firebase", &projects.ServiceArgs{
Project: _default.ProjectId,
Service: pulumi.String("firebase.googleapis.com"),
})
if err != nil {
return err
}
defaultProject, err := firebase.NewProject(ctx, "default", &firebase.ProjectArgs{
Project: _default.ProjectId,
}, pulumi.DependsOn([]pulumi.Resource{
firebase,
}))
if err != nil {
return err
}
firebaseDatabase, err := projects.NewService(ctx, "firebase_database", &projects.ServiceArgs{
Project: defaultProject.Project,
Service: pulumi.String("firebasedatabase.googleapis.com"),
})
if err != nil {
return err
}
wait60Seconds, err := time.NewSleep(ctx, "wait_60_seconds", &time.SleepArgs{
CreateDuration: pulumi.String("60s"),
}, pulumi.DependsOn([]pulumi.Resource{
firebaseDatabase,
}))
if err != nil {
return err
}
_, err = firebase.NewDatabaseInstance(ctx, "default", &firebase.DatabaseInstanceArgs{
Project: defaultProject.Project,
Region: pulumi.String("us-central1"),
InstanceId: pulumi.String("rtdb-project-default-rtdb"),
Type: pulumi.String("DEFAULT_DATABASE"),
}, pulumi.DependsOn([]pulumi.Resource{
wait60Seconds,
}))
if err != nil {
return err
}
return nil
})
}
using System.Collections.Generic;
using System.Linq;
using Pulumi;
using Gcp = Pulumi.Gcp;
using Time = Pulumiverse.Time;
return await Deployment.RunAsync(() =>
{
var @default = new Gcp.Organizations.Project("default", new()
{
ProjectId = "rtdb-project",
Name = "rtdb-project",
OrgId = "123456789",
DeletionPolicy = "DELETE",
Labels =
{
{ "firebase", "enabled" },
},
});
var firebase = new Gcp.Projects.Service("firebase", new()
{
Project = @default.ProjectId,
ServiceName = "firebase.googleapis.com",
});
var defaultProject = new Gcp.Firebase.Project("default", new()
{
ProjectID = @default.ProjectId,
}, new CustomResourceOptions
{
DependsOn =
{
firebase,
},
});
var firebaseDatabase = new Gcp.Projects.Service("firebase_database", new()
{
Project = defaultProject.ProjectID,
ServiceName = "firebasedatabase.googleapis.com",
});
var wait60Seconds = new Time.Sleep("wait_60_seconds", new()
{
CreateDuration = "60s",
}, new CustomResourceOptions
{
DependsOn =
{
firebaseDatabase,
},
});
var defaultDatabaseInstance = new Gcp.Firebase.DatabaseInstance("default", new()
{
Project = defaultProject.ProjectID,
Region = "us-central1",
InstanceId = "rtdb-project-default-rtdb",
Type = "DEFAULT_DATABASE",
}, new CustomResourceOptions
{
DependsOn =
{
wait60Seconds,
},
});
});
package generated_program;
import com.pulumi.Context;
import com.pulumi.Pulumi;
import com.pulumi.core.Output;
import com.pulumi.gcp.projects.Service;
import com.pulumi.gcp.projects.ServiceArgs;
import com.pulumiverse.time.Sleep;
import com.pulumiverse.time.SleepArgs;
import com.pulumi.gcp.firebase.DatabaseInstance;
import com.pulumi.gcp.firebase.DatabaseInstanceArgs;
import com.pulumi.resources.CustomResourceOptions;
import java.util.List;
import java.util.ArrayList;
import java.util.Map;
import java.io.File;
import java.nio.file.Files;
import java.nio.file.Paths;
public class App {
public static void main(String[] args) {
Pulumi.run(App::stack);
}
public static void stack(Context ctx) {
var default_ = new com.pulumi.gcp.organizations.Project("default", com.pulumi.gcp.organizations.ProjectArgs.builder()
.projectId("rtdb-project")
.name("rtdb-project")
.orgId("123456789")
.deletionPolicy("DELETE")
.labels(Map.of("firebase", "enabled"))
.build());
var firebase = new Service("firebase", ServiceArgs.builder()
.project(default_.projectId())
.service("firebase.googleapis.com")
.build());
var defaultProject = new com.pulumi.gcp.firebase.Project("defaultProject", com.pulumi.gcp.firebase.ProjectArgs.builder()
.project(default_.projectId())
.build(), CustomResourceOptions.builder()
.dependsOn(firebase)
.build());
var firebaseDatabase = new Service("firebaseDatabase", ServiceArgs.builder()
.project(defaultProject.project())
.service("firebasedatabase.googleapis.com")
.build());
var wait60Seconds = new Sleep("wait60Seconds", SleepArgs.builder()
.createDuration("60s")
.build(), CustomResourceOptions.builder()
.dependsOn(firebaseDatabase)
.build());
var defaultDatabaseInstance = new DatabaseInstance("defaultDatabaseInstance", DatabaseInstanceArgs.builder()
.project(defaultProject.project())
.region("us-central1")
.instanceId("rtdb-project-default-rtdb")
.type("DEFAULT_DATABASE")
.build(), CustomResourceOptions.builder()
.dependsOn(wait60Seconds)
.build());
}
}
resources:
default:
type: gcp:organizations:Project
properties:
projectId: rtdb-project
name: rtdb-project
orgId: '123456789'
deletionPolicy: DELETE
labels:
firebase: enabled
firebase:
type: gcp:projects:Service
properties:
project: ${default.projectId}
service: firebase.googleapis.com
defaultProject:
type: gcp:firebase:Project
name: default
properties:
project: ${default.projectId}
options:
dependsOn:
- ${firebase}
firebaseDatabase:
type: gcp:projects:Service
name: firebase_database
properties:
project: ${defaultProject.project}
service: firebasedatabase.googleapis.com
wait60Seconds:
type: time:Sleep
name: wait_60_seconds
properties:
createDuration: 60s
options:
dependsOn:
- ${firebaseDatabase}
defaultDatabaseInstance:
type: gcp:firebase:DatabaseInstance
name: default
properties:
project: ${defaultProject.project}
region: us-central1
instanceId: rtdb-project-default-rtdb
type: DEFAULT_DATABASE
options:
dependsOn:
- ${wait60Seconds}
The type property set to DEFAULT_DATABASE creates the project’s immutable default database. The dependsOn chain ensures APIs are enabled before database creation. The 60-second wait accounts for API propagation delays across Google’s infrastructure. Default databases use a special naming convention: {project-id}-default-rtdb.
Beyond these examples
These snippets focus on specific database instance features: user and default database types, regional placement, and state management. They’re intentionally minimal rather than full Firebase deployments.
The examples assume pre-existing infrastructure such as a GCP project with Firebase enabled, the firebasedatabase.googleapis.com API activated, and a Blaze plan for user databases. They focus on provisioning the database instance rather than configuring security rules or application integration.
To keep things focused, common database patterns are omitted, including:
- Security rules configuration
- Database URL usage in application code
- Backup and restore operations
- Multi-region replication strategies
These omissions are intentional: the goal is to illustrate how each database instance feature is wired, not provide drop-in Firebase modules. See the Firebase DatabaseInstance resource reference for all available configuration options.
Let's deploy GCP Firebase Realtime Database Instances
Get started with Pulumi Cloud, then follow our quick setup guide to deploy this infrastructure.
Try Pulumi Cloud for FREEFrequently Asked Questions
Database Types & Limitations
DEFAULT_DATABASE is a single database per project that cannot be deleted once created (only disabled). USER_DATABASE is the default type, allows multiple databases, and can be deleted, but requires a Blaze plan.type set to DEFAULT_DATABASE cannot be deleted once created. You can only disable them by setting desiredState to DISABLED.USER_DATABASE instances requires a Blaze (pay-as-you-go) plan. Projects can be upgraded using the Cloud Billing API.desiredState to DISABLED. This works for both DEFAULT_DATABASE and USER_DATABASE types. Set it back to ACTIVE to reenable.Instance IDs & Immutability
instanceId, project, region, and type properties are all immutable and require replacing the resource to change.instanceId when creating a replacement database.region property is immutable after creation.Setup & Dependencies
time.Sleep with a 60-second duration after enabling the firebasedatabase.googleapis.com service, indicating that API propagation time may be needed before creating the database instance.us-central1 instances, the URL is https://{instance-id}.firebaseio.com. For other regions, it’s https://{instance-id}.{region}.firebasedatabase.app.