Skip to content

Commit

Permalink
feat(bigquery): add support for storage billing model (#7510)
Browse files Browse the repository at this point in the history
Resolves #6978
  • Loading branch information
alvarowolfx authored Mar 9, 2023
1 parent 97337e9 commit 0132ca9
Show file tree
Hide file tree
Showing 3 changed files with 110 additions and 4 deletions.
28 changes: 27 additions & 1 deletion bigquery/dataset.go
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,12 @@ type DatasetMetadata struct {
// More information: https://cloud.google.com/bigquery/docs/reference/standard-sql/collation-concepts
DefaultCollation string

// Storage billing model to be used for all tables in the dataset.
// Can be set to PHYSICAL. Default is LOGICAL.
// Once you create a dataset with storage billing model set to physical bytes, you can't change it back to using logical bytes again.
// More details: https://cloud.google.com/bigquery/docs/datasets-intro#dataset_storage_billing_models
StorageBillingModel string

// These fields are read-only.
CreationTime time.Time
LastModifiedTime time.Time // When the dataset or any of its tables were modified.
Expand All @@ -84,6 +90,14 @@ type DatasetTag struct {
TagValue string
}

const (
// LogicalStorageBillingModel indicates billing for logical bytes.
LogicalStorageBillingModel = ""

// PhysicalStorageBillingModel indicates billing for physical bytes.
PhysicalStorageBillingModel = "PHYSICAL"
)

func bqToDatasetTag(in *bq.DatasetTags) *DatasetTag {
if in == nil {
return nil
Expand Down Expand Up @@ -117,6 +131,12 @@ type DatasetMetadataToUpdate struct {
// created in the dataset.
DefaultCollation optional.String

// Storage billing model to be used for all tables in the dataset.
// Can be set to PHYSICAL. Default is LOGICAL.
// Once you change a dataset's storage billing model to use physical bytes, you can't change it back to using logical bytes again.
// More details: https://cloud.google.com/bigquery/docs/datasets-intro#dataset_storage_billing_models
StorageBillingModel optional.String

// The entire access list. It is not possible to replace individual entries.
Access []*AccessEntry

Expand Down Expand Up @@ -187,7 +207,8 @@ func (dm *DatasetMetadata) toBQ() (*bq.Dataset, error) {
ds.Location = dm.Location
ds.DefaultTableExpirationMs = int64(dm.DefaultTableExpiration / time.Millisecond)
ds.DefaultPartitionExpirationMs = int64(dm.DefaultPartitionExpiration / time.Millisecond)
ds.DefaultCollation = string(dm.DefaultCollation)
ds.DefaultCollation = dm.DefaultCollation
ds.StorageBillingModel = string(dm.StorageBillingModel)
ds.Labels = dm.Labels
var err error
ds.Access, err = accessListToBQ(dm.Access)
Expand Down Expand Up @@ -274,6 +295,7 @@ func bqToDatasetMetadata(d *bq.Dataset, c *Client) (*DatasetMetadata, error) {
DefaultTableExpiration: time.Duration(d.DefaultTableExpirationMs) * time.Millisecond,
DefaultPartitionExpiration: time.Duration(d.DefaultPartitionExpirationMs) * time.Millisecond,
DefaultCollation: d.DefaultCollation,
StorageBillingModel: d.StorageBillingModel,
DefaultEncryptionConfig: bqToEncryptionConfig(d.DefaultEncryptionConfiguration),
Description: d.Description,
Name: d.FriendlyName,
Expand Down Expand Up @@ -363,6 +385,10 @@ func (dm *DatasetMetadataToUpdate) toBQ() (*bq.Dataset, error) {
ds.DefaultCollation = optional.ToString(dm.DefaultCollation)
forceSend("DefaultCollation")
}
if dm.StorageBillingModel != nil {
ds.StorageBillingModel = optional.ToString(dm.StorageBillingModel)
forceSend("StorageBillingModel")
}
if dm.DefaultEncryptionConfig != nil {
ds.DefaultEncryptionConfiguration = dm.DefaultEncryptionConfig.toBQ()
ds.DefaultEncryptionConfiguration.ForceSendFields = []string{"KmsKeyName"}
Expand Down
77 changes: 77 additions & 0 deletions bigquery/dataset_integration_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -277,6 +277,83 @@ func TestIntegration_DatasetUpdateDefaultCollation(t *testing.T) {
}
}

func TestIntegration_DatasetStorageBillingModel(t *testing.T) {
if client == nil {
t.Skip("Integration tests skipped")
}

ctx := context.Background()
md, err := dataset.Metadata(ctx)
if err != nil {
t.Fatal(err)
}
if md.StorageBillingModel != LogicalStorageBillingModel {
t.Fatalf("got %q, want %q", md.StorageBillingModel, LogicalStorageBillingModel)
}

ds := client.Dataset(datasetIDs.New())
err = ds.Create(ctx, &DatasetMetadata{
StorageBillingModel: PhysicalStorageBillingModel,
})
if err != nil {
t.Fatal(err)
}
md, err = ds.Metadata(ctx)
if err != nil {
t.Fatal(err)
}
if md.StorageBillingModel != PhysicalStorageBillingModel {
t.Fatalf("got %q, want %q", md.StorageBillingModel, PhysicalStorageBillingModel)
}
if err := ds.Delete(ctx); err != nil {
t.Fatalf("deleting dataset %v: %v", ds, err)
}
}

func TestIntegration_DatasetUpdateStorageBillingModel(t *testing.T) {
if client == nil {
t.Skip("Integration tests skipped")
}

ctx := context.Background()
ds := client.Dataset(datasetIDs.New())
err := ds.Create(ctx, &DatasetMetadata{
StorageBillingModel: LogicalStorageBillingModel,
})
if err != nil {
t.Fatal(err)
}

md, err := ds.Metadata(ctx)
if md.StorageBillingModel != LogicalStorageBillingModel {
t.Fatalf("got %q, want %q", md.StorageBillingModel, LogicalStorageBillingModel)
}

// Update the Storage billing model
md, err = ds.Update(ctx, DatasetMetadataToUpdate{
StorageBillingModel: PhysicalStorageBillingModel,
}, "")
if err != nil {
t.Fatal(err)
}
if md.StorageBillingModel != PhysicalStorageBillingModel {
t.Fatalf("got %q, want %q", md.StorageBillingModel, PhysicalStorageBillingModel)
}

// Omitting StorageBillingModel doesn't change it.
md, err = ds.Update(ctx, DatasetMetadataToUpdate{Name: "xyz"}, "")
if err != nil {
t.Fatal(err)
}
if md.StorageBillingModel != PhysicalStorageBillingModel {
t.Fatalf("got %q, want %q", md.StorageBillingModel, PhysicalStorageBillingModel)
}

if err := ds.Delete(ctx); err != nil {
t.Fatalf("deleting dataset %v: %v", ds, err)
}
}

func TestIntegration_DatasetUpdateAccess(t *testing.T) {
if client == nil {
t.Skip("Integration tests skipped")
Expand Down
9 changes: 6 additions & 3 deletions bigquery/dataset_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -432,8 +432,9 @@ func TestBQToDatasetMetadata(t *testing.T) {
DefaultEncryptionConfig: &EncryptionConfig{
KMSKeyName: "some_key",
},
Location: "EU",
Labels: map[string]string{"x": "y"},
StorageBillingModel: LogicalStorageBillingModel,
Location: "EU",
Labels: map[string]string{"x": "y"},
Access: []*AccessEntry{
{Role: ReaderRole, Entity: "[email protected]", EntityType: UserEmailEntity},
{Role: WriterRole, Entity: "[email protected]", EntityType: GroupEmailEntity},
Expand Down Expand Up @@ -466,6 +467,7 @@ func TestDatasetMetadataToUpdateToBQ(t *testing.T) {
Name: "name",
DefaultTableExpiration: time.Hour,
DefaultPartitionExpiration: 24 * time.Hour,
StorageBillingModel: PhysicalStorageBillingModel,
DefaultEncryptionConfig: &EncryptionConfig{
KMSKeyName: "some_key",
},
Expand All @@ -482,12 +484,13 @@ func TestDatasetMetadataToUpdateToBQ(t *testing.T) {
FriendlyName: "name",
DefaultTableExpirationMs: 60 * 60 * 1000,
DefaultPartitionExpirationMs: 24 * 60 * 60 * 1000,
StorageBillingModel: string(PhysicalStorageBillingModel),
DefaultEncryptionConfiguration: &bq.EncryptionConfiguration{
KmsKeyName: "some_key",
ForceSendFields: []string{"KmsKeyName"},
},
Labels: map[string]string{"label": "value"},
ForceSendFields: []string{"Description", "FriendlyName"},
ForceSendFields: []string{"Description", "FriendlyName", "StorageBillingModel"},
NullFields: []string{"Labels.del"},
}
if diff := testutil.Diff(got, want); diff != "" {
Expand Down

0 comments on commit 0132ca9

Please sign in to comment.