pg-proto-parser is a TypeScript project that parses pganalyze/libpg_query PostgreSQL Protocol Buffers (protobuf) definitions and generates TypeScript interfaces, utility functions, and JSON mappings for the enums defined in the protobuf schema. Designed to work with launchql/pgsql-parser for maintainable upgrades.
- Parses protobuf definitions and creates a structured representation in TypeScript.
- Generates TypeScript interfaces for protobuf messages.
- Creates utility functions for enum value conversions with multiple output formats:
- Bidirectional conversion functions (string ↔ number)
- Unidirectional conversion functions with precise types
- Switch statement or nested object output formats
- Produces JSON or TypeScript files mapping enum names to integer values and vice versa
- Supports tree-shakable enum utilities for optimal bundle sizes
Here's how to parse protobuf files and generate the output:
import { PgProtoParser } from 'pg-proto-parser';
// Create PgProtoParser
const parser = new PgProtoParser(inFile, { outDir });
// Generate TypeScript and JSON files
await parser.write();pg-proto-parser will generate the following files in the specified outDir:
types.ts: TypeScript file containing interfaces for protobuf messages.enums.ts: TypeScript file containing enums for protobuf messages.asts.ts: TypeScript file containing helpers to create PostgreSQL ASTs.utils.ts: TypeScript file containing utility functions for enums.enums2int.json: JSON mapping of enum names to integer values.enums2str.json: JSON mapping of integer values to enum names.
You can generate TypeScript exports instead of JSON files for enum mappings:
const parser = new PgProtoParser(inFile, {
outDir,
enums: {
enumMap: {
enabled: true,
format: 'ts', // Generate TypeScript exports
toIntOutFile: 'enum-to-int.ts',
toStrOutFile: 'enum-to-str.ts'
}
}
});
await parser.write();This will generate TypeScript files with exports like:
// enum-to-int.ts
export const enumToIntMap = {
OverridingKind: {
OVERRIDING_NOT_SET: 0,
OVERRIDING_USER_VALUE: 1,
OVERRIDING_SYSTEM_VALUE: 2
},
// ... more enums
};
export type EnumToIntMap = typeof enumToIntMap;
// enum-to-str.ts
export const enumToStrMap = {
OverridingKind: {
"0": 'OVERRIDING_NOT_SET',
"1": 'OVERRIDING_USER_VALUE',
"2": 'OVERRIDING_SYSTEM_VALUE'
},
// ... more enums
};
export type EnumToStrMap = typeof enumToStrMap;You can generate utility functions for runtime enum conversions:
// Bidirectional function (default)
const parser = new PgProtoParser(inFile, {
outDir,
utils: {
enums: {
enabled: true,
filename: 'enum-utils.ts'
}
}
});
// Unidirectional functions with switch statements
const parser = new PgProtoParser(inFile, {
outDir,
utils: {
enums: {
enabled: true,
unidirectional: true,
toIntFilename: 'enum-to-int.ts',
toStringFilename: 'enum-to-string.ts'
}
}
});
// Unidirectional functions with nested objects format
const parser = new PgProtoParser(inFile, {
outDir,
utils: {
enums: {
enabled: true,
unidirectional: true,
outputFormat: 'nestedObjects',
toIntFilename: 'enum-to-int-map.ts',
toStringFilename: 'enum-to-string-map.ts'
}
}
});Bidirectional (default):
// utils.ts
export const getEnumValue = (enumType: EnumType, key: string | number) => {
// Returns number for string input, string for number input
};Unidirectional with switch statements:
// enum-to-int.ts
export const getEnumInt = (enumType: EnumType, key: string): number => {
// Converts enum string to number
};
// enum-to-string.ts
export const getEnumString = (enumType: EnumType, key: number): string => {
// Converts enum number to string
};Unidirectional with nested objects:
// enum-to-int-map.ts
export const enumToIntMap = {
OverridingKind: (key: string): number => { /* ... */ },
QuerySource: (key: string): number => { /* ... */ },
// ... more enums
};
// Usage
const value = enumToIntMap.OverridingKind("OVERRIDING_USER_VALUE"); // Returns: 1You can configure pg-proto-parser by passing different parameters to the ProtoStore constructor:
root: The protobufRootobject containing your schema.options: Options defined asPgProtoParserOptions(see below)
The options for PgProtoParserOptions are organized into the following categories:
| Option | Description | Default Value |
|---|---|---|
outDir |
The directory where the generated files will be saved. | process.cwd() + "/out" |
exclude |
List of type or enum names to exclude during processing. | [] |
| Option | Description | Default Value |
|---|---|---|
types.enabled |
Whether to generate TypeScript interfaces for protobuf messages. | false |
types.filename |
Filename for the generated TypeScript interfaces. | 'types.ts' |
types.optionalFields |
Generates TypeScript interfaces with optional fields mapping to the PostgreSQL node types' fields; sets all fields to optional. | true |
types.enumsSource |
Path to the TypeScript enums to use when generating TypeScript interfaces. | './enums' |
types.wrappedNodeTypeExport |
Simpler option for exporting Node type references. | true |
| Option | Description | Default Value |
|---|---|---|
enums.enabled |
Outputs TypeScript enum types for the PostgreSQL enums. | false |
enums.filename |
Filename for the generated TypeScript enums. | 'enums.ts' |
enums.enumsAsTypeUnion |
Uses strings to define enum types as specified for the fields of each proto message type. | true |
enums.removeUndefinedAt0 |
Removes the initial UNDEFINED enum entry and adjusts the subsequent values by decrementing them. |
true |
| Option | Description | Default Value |
|---|---|---|
enums.enumMap.enabled |
Whether to generate enum mapping files (replaces the old enums.json options). |
false |
enums.enumMap.format |
Output format for enum mappings: 'json' for plain JSON files, 'ts' for TypeScript exports. | 'json' |
enums.enumMap.toIntOutFile |
Output file name for the mapping of enum names to integer values. | 'enums2int.json' |
enums.enumMap.toStrOutFile |
Output file name for the mapping of integer values to enum names. | 'enums2str.json' |
Note: The enums.json.* options are deprecated in favor of enums.enumMap.*. When using TypeScript format (format: 'ts'), the file extensions will be automatically corrected to .ts.
| Option | Description | Default Value |
|---|---|---|
utils.enums.enabled |
Whether to generate TypeScript utility functions for enums. | false |
utils.enums.filename |
Filename for the generated enums utilities (bidirectional). | 'utils.ts' |
utils.enums.unidirectional |
Generate separate unidirectional conversion functions instead of bidirectional. | false |
utils.enums.toIntFilename |
Filename for string-to-int conversion utilities (when unidirectional). | 'enum-to-int.ts' |
utils.enums.toStringFilename |
Filename for int-to-string conversion utilities (when unidirectional). | 'enum-to-string.ts' |
utils.enums.outputFormat |
Output format: 'switchStatements' or 'nestedObjects'. | 'switchStatements' |
utils.astHelpers.enabled |
Outputs TypeScript helpers for building PostgreSQL ASTs. | false |
utils.astHelpers.wrappedTypesSource |
Path to the TypeScript types to use when generating AST helpers. | './wrapped' |
utils.astHelpers.inlineNestedObj |
Whether to inline nested-obj code within the generated file. |
false |
utils.astHelpers.nestedObjFile |
Filename for the inlined nested-obj code, if inlineNestedObj is true. |
'nested-obj.ts' |
utils.astHelpers.filename |
Filename for the generated AST helpers. | 'asts.ts' |
| Option | Description | Default Value |
|---|---|---|
runtimeSchema.enabled |
Whether to generate runtime schema for AST nodes with metadata about node types and field specifications. | false |
runtimeSchema.filename |
Filename for the generated runtime schema file. | 'runtime-schema' |
runtimeSchema.format |
Output format for runtime schema generation ('json' or 'typescript'). | 'json' |
Each of these options can be set when initializing the PgProtoParser to customize its behavior and output.
generateTsAstCodeFromPgAst is a method that transforms a PostgreSQL Abstract Syntax Tree (AST) into TypeScript code capable of generating an equivalent AST. This function facilitates the dynamic creation of ASTs, allowing for programmable query construction and manipulation in TypeScript.
It generates code with syntax for @pgsql/utils, using runtime schema information to determine whether nodes should be wrapped (t.nodes.*) or unwrapped (t.ast.*).
import { generateTsAstCodeFromPgAst } from 'pg-proto-parser';
import { runtimeSchema } from './path/to/your/generated/runtime-schema';
import { parse } from 'pgsql-parser';
// Example SQL query
const sql = 'SELECT * FROM my_table WHERE id = 1';
// Parse the SQL query to get the PostgreSQL AST
const pgAst = parse(sql);
// Generate TypeScript AST builder code from the PostgreSQL AST using runtime schema
const tsAstBuilderCode = generateTsAstCodeFromPgAst(
pgAst[0].RawStmt.stmt,
runtimeSchema
);
console.log(tsAstBuilderCode);
// OUTPUT BELOW:
// make sure to use this import when using
// import ast from '@pgsql/utils';
t.nodes.selectStmt({
targetList: [t.nodes.resTarget({
val: t.nodes.columnRef({
fields: [t.nodes.aStar({})],
location: 7
}),
location: 7
})],
fromClause: [t.nodes.rangeVar({
relname: "my_table",
inh: true,
relpersistence: "p",
location: 14
})],
whereClause: t.nodes.aExpr({
kind: "AEXPR_OP",
name: [t.nodes.string({
sval: "="
})],
lexpr: t.nodes.columnRef({
fields: [t.nodes.string({
sval: "id"
})],
location: 29
}),
rexpr: t.nodes.aConst({
ival: t.ast.integer({
ival: 1
}),
location: 34
}),
location: 32
}),
limitOption: "LIMIT_OPTION_DEFAULT",
op: "SETOP_NONE"
})