Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Declaration configs as protos #1767

Merged
merged 1 commit into from
Jun 26, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Declaration configs as protos
  • Loading branch information
Ekrekr committed Jun 26, 2024
commit a2022eabc5d6a7d526e0e6bfdb58c500a9e3ed7e
90 changes: 53 additions & 37 deletions core/actions/declaration.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,26 +8,15 @@ import {
ITargetableConfig
} from "df/core/common";
import { Session } from "df/core/session";
import {
actionConfigToCompiledGraphTarget,
checkExcessProperties,
strictKeysOf
} from "df/core/utils";
import { actionConfigToCompiledGraphTarget, checkExcessProperties } from "df/core/utils";
import { dataform } from "df/protos/ts";

/**
* Configuration options for `declaration` action types.
*/
export interface IDeclarationConfig extends IDocumentableConfig, INamedConfig, ITargetableConfig {}

export const IDeclarationConfigProperties = strictKeysOf<IDeclarationConfig>()([
"columns",
"database",
"description",
"name",
"schema",
"type"
]);
interface ILegacyDeclarationConfig extends dataform.ActionConfig.DeclarationConfig {
database: string;
schema: string;
fileName: string;
type: string;
}

/**
* @hidden
Expand All @@ -38,14 +27,16 @@ export class Declaration extends ActionBuilder<dataform.Declaration> {

public session: Session;

constructor(session?: Session, config?: dataform.ActionConfig.DeclarationConfig) {
constructor(session?: Session, unverifiedConfig?: any) {
super(session);
this.session = session;

if (!config) {
if (!unverifiedConfig) {
return;
}

const config = this.verifyConfig(unverifiedConfig);

if (!config.name) {
throw Error("Declarations must have a populated 'name' field.");
}
Expand All @@ -54,22 +45,15 @@ export class Declaration extends ActionBuilder<dataform.Declaration> {
this.proto.target = this.applySessionToTarget(target, session.projectConfig);
this.proto.canonicalTarget = this.applySessionToTarget(target, session.canonicalProjectConfig);

// TODO(ekrekr): load config proto column descriptors.
this.config({ description: config.description });
}

public config(config: IDeclarationConfig) {
checkExcessProperties(
(e: Error) => this.session.compileError(e),
config,
IDeclarationConfigProperties,
"declaration config"
);
if (config.description) {
this.description(config.description);
}
if (config.columns) {
this.columns(config.columns);
if (config.columns?.length) {
this.columns(
config.columns.map(columnDescriptor =>
dataform.ActionConfig.ColumnDescriptor.create(columnDescriptor)
)
);
}
return this;
}
Expand All @@ -82,13 +66,12 @@ export class Declaration extends ActionBuilder<dataform.Declaration> {
return this;
}

public columns(columns: IColumnsDescriptor) {
public columns(columns: dataform.ActionConfig.ColumnDescriptor[]) {
if (!this.proto.actionDescriptor) {
this.proto.actionDescriptor = {};
}
this.proto.actionDescriptor.columns = ColumnDescriptors.mapToColumnProtoArray(
columns,
(e: Error) => this.session.compileError(e)
this.proto.actionDescriptor.columns = ColumnDescriptors.mapConfigProtoToCompilationProto(
columns
);
return this;
}
Expand All @@ -114,4 +97,37 @@ export class Declaration extends ActionBuilder<dataform.Declaration> {
VerifyProtoErrorBehaviour.SUGGEST_REPORTING_TO_DATAFORM_TEAM
);
}

private verifyConfig(
unverifiedConfig: ILegacyDeclarationConfig
): dataform.ActionConfig.DeclarationConfig {
if (unverifiedConfig.database) {
unverifiedConfig.project = unverifiedConfig.database;
delete unverifiedConfig.database;
}
if (unverifiedConfig.schema) {
unverifiedConfig.dataset = unverifiedConfig.schema;
delete unverifiedConfig.schema;
}
if (unverifiedConfig.columns) {
// TODO(ekrekr) columns in their current config format are a difficult structure to represent
// as protos. They are nested, and use the object keys as the names. Consider a forced
// migration to the proto style column names.
unverifiedConfig.columns = ColumnDescriptors.mapLegacyObjectToConfigProto(
unverifiedConfig.columns as any
);
}

// TODO(ekrekr): consider moving this to a shared location after all action builders have proto
// config verifiers.
if (unverifiedConfig.type) {
delete unverifiedConfig.type;
}

return verifyObjectMatchesProto(
dataform.ActionConfig.DeclarationConfig,
unverifiedConfig,
VerifyProtoErrorBehaviour.SHOW_DOCS_LINK
);
}
}
4 changes: 2 additions & 2 deletions core/actions/operation.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { verifyObjectMatchesProto, VerifyProtoErrorBehaviour } from "df/common/protos";
import { ActionBuilder } from "df/core/actions";
import { ColumnDescriptors } from "df/core/column_descriptors";
import { LegacyColumnDescriptors } from "df/core/column_descriptors";
import {
Contextable,
IActionConfig,
Expand Down Expand Up @@ -213,7 +213,7 @@ export class Operation extends ActionBuilder<dataform.Operation> {
if (!this.proto.actionDescriptor) {
this.proto.actionDescriptor = {};
}
this.proto.actionDescriptor.columns = ColumnDescriptors.mapToColumnProtoArray(
this.proto.actionDescriptor.columns = LegacyColumnDescriptors.mapToColumnProtoArray(
columns,
(e: Error) => this.session.compileError(e)
);
Expand Down
4 changes: 2 additions & 2 deletions core/actions/table.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { verifyObjectMatchesProto, VerifyProtoErrorBehaviour } from "df/common/protos";
import { ActionBuilder } from "df/core/actions";
import { Assertion } from "df/core/actions/assertion";
import { ColumnDescriptors } from "df/core/column_descriptors";
import { LegacyColumnDescriptors } from "df/core/column_descriptors";
import {
Contextable,
IActionConfig,
Expand Down Expand Up @@ -606,7 +606,7 @@ export class Table extends ActionBuilder<dataform.Table> {
if (!this.proto.actionDescriptor) {
this.proto.actionDescriptor = {};
}
this.proto.actionDescriptor.columns = ColumnDescriptors.mapToColumnProtoArray(
this.proto.actionDescriptor.columns = LegacyColumnDescriptors.mapToColumnProtoArray(
columns,
(e: Error) => this.session.compileError(e)
);
Expand Down
68 changes: 66 additions & 2 deletions core/column_descriptors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,77 @@ import { dataform } from "df/protos/ts";
* @hidden
*/
export class ColumnDescriptors {
public static mapConfigProtoToCompilationProto(
columns: dataform.ActionConfig.ColumnDescriptor[]
): dataform.IColumnDescriptor[] {
return columns.map(column => {
return dataform.ColumnDescriptor.create({
path: column.path,
description: column.description,
tags: column.tags,
bigqueryPolicyTags: column.bigqueryPolicyTags
});
});
}

public static mapLegacyObjectToConfigProto(
columns: IColumnsDescriptor
): dataform.ActionConfig.ColumnDescriptor[] {
return Object.keys(columns)
.map(column => ColumnDescriptors.mapColumnDescriptionToProto([column], columns[column]))
.flat();
}

public static mapColumnDescriptionToProto(
currentPath: string[],
description: string | IRecordDescriptor
): dataform.ActionConfig.ColumnDescriptor[] {
if (typeof description === "string") {
return [
dataform.ColumnDescriptor.create({
description,
path: currentPath
})
];
}
const columnDescriptor: dataform.ActionConfig.ColumnDescriptor[] = !!description
? [
dataform.ActionConfig.ColumnDescriptor.create({
path: currentPath,
description: description.description,
tags: typeof description.tags === "string" ? [description.tags] : description.tags,
bigqueryPolicyTags:
typeof description.bigqueryPolicyTags === "string"
? [description.bigqueryPolicyTags]
: description.bigqueryPolicyTags
})
]
: [];
const nestedColumns = description.columns ? Object.keys(description.columns) : [];
return columnDescriptor.concat(
nestedColumns
.map(nestedColumn =>
ColumnDescriptors.mapColumnDescriptionToProto(
currentPath.concat([nestedColumn]),
description.columns[nestedColumn]
)
)
.flat()
);
}
}

/**
* @hidden
*/
export class LegacyColumnDescriptors {
public static mapToColumnProtoArray(
columns: IColumnsDescriptor,
reportError: (e: Error) => void
): dataform.IColumnDescriptor[] {
return Object.keys(columns)
.map(column =>
ColumnDescriptors.mapColumnDescriptionToProto([column], columns[column], reportError)
LegacyColumnDescriptors.mapColumnDescriptionToProto([column], columns[column], reportError)
)
.flat();
}
Expand Down Expand Up @@ -54,7 +118,7 @@ export class ColumnDescriptors {
return columnDescriptor.concat(
nestedColumns
.map(nestedColumn =>
ColumnDescriptors.mapColumnDescriptionToProto(
LegacyColumnDescriptors.mapColumnDescriptionToProto(
currentPath.concat([nestedColumn]),
description.columns[nestedColumn],
reportError
Expand Down
2 changes: 0 additions & 2 deletions core/main_test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1345,7 +1345,6 @@ actions:
columns: {
nestedColumnKey: "nestedColumnVal"
},
displayName: "displayName",
tags: ["tag3", "tag4"],
bigqueryPolicyTags: ["bigqueryPolicyTag1", "bigqueryPolicyTag2"],
}
Expand All @@ -1359,7 +1358,6 @@ actions:
{
bigqueryPolicyTags: ["bigqueryPolicyTag1", "bigqueryPolicyTag2"],
description: "description",
displayName: "displayName",
path: ["column2Key"],
tags: ["tag3", "tag4"]
},
Expand Down
8 changes: 3 additions & 5 deletions core/session.ts
Original file line number Diff line number Diff line change
Expand Up @@ -161,11 +161,9 @@ export class Session {
.queries(actionOptions.sqlContextable);
break;
case "declaration":
this.declare({
database: sqlxConfig.database,
schema: sqlxConfig.schema,
name: sqlxConfig.name
}).config(sqlxConfig);
const declaration = new Declaration(this, sqlxConfig);
declaration.proto.fileName = utils.getCallerFile(this.rootDir);
this.actions.push(declaration);
break;
case "test":
const testCase = this.test(sqlxConfig.name)
Expand Down
3 changes: 3 additions & 0 deletions protos/configs.proto
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,9 @@ message ActionConfig {

// A list of BigQuery policy tags that will be applied to the column.
repeated string bigquery_policy_tags = 3;

// A list of tags for this column which will be applied.
repeated string tags = 4;
}

message TableConfig {
Expand Down
Loading