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

[Beyonce]: Allow Compound Partition/Sort Keys #34

Merged
merged 5 commits into from
Jul 20, 2020
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
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ Tables:

#### A note on `partitionKey` and `sortKey` syntax

Beyonce expects you to specify your partition and sort keys as tuple-2s, e.g. `[Author, $id]`. The first element is a "key prefix" and the 2nd must be a field on your model. For example we set the primary key of the `Author` model above to: `[Author, $id]`, would result in the key: `Author-$id`, where `$id` is the value of a specific Author's id.
Beyonce expects you to specify your partition and sort keys as arrays, e.g. `[Author, $id]`. The first element is a "key prefix" and all subsequent items must be field names on your model. For example we set the primary key of the `Author` model above to: `[Author, $id]`, would result in the key: `Author-$id`, where `$id` is the value of a specific Author's id. And if we wanted a compound key we could do `[Author, $id, $name]`. You can specify compound keys for both partition and sort keys.

Using the example above, if we wanted to place `Books` under the same partition key, then we'd need to set the `Book` model's `partitionKey` to `[Author, $authorId]`.

Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@ginger.io/beyonce",
"version": "0.0.37",
"version": "0.0.38",
"description": "Type-safe DynamoDB query builder for TypeScript. Designed with single-table architecture in mind.",
"main": "dist/index.js",
"types": "dist/index.d.ts",
Expand Down
12 changes: 8 additions & 4 deletions src/main/codegen/generateModels.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,20 @@
import { Model } from "./types"
import { formatKeyComponent } from "./util"

export function generateModels(models: Model[]) {
return models.map(generateModel)
}

function generateModel(model: Model): string {
const [pkPrefix, pk] = model.keys.partitionKey
const [skPrefix, sk] = model.keys.sortKey
const [pkPrefix, ...pkComponents] = model.keys.partitionKey.map(
formatKeyComponent
)

const [skPrefix, ...skComponents] = model.keys.sortKey.map(formatKeyComponent)

return `export const ${model.name}Model = ${model.tableName}Table
.model<${model.name}>(ModelType.${model.name})
.partitionKey("${pkPrefix}", "${pk.replace("$", "")}")
.sortKey("${skPrefix}", "${sk.replace("$", "")}")
.partitionKey(${pkPrefix}, ${pkComponents.join(", ")})
.sortKey(${skPrefix}, ${skComponents.join(", ")})
`
}
23 changes: 15 additions & 8 deletions src/main/codegen/generateTables.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { Table } from "./types"
import { formatKeyComponent } from "./util"

export function generateTables(tables: Table[]): string {
const tableCode = tables.map((table) => {
Expand All @@ -22,16 +23,22 @@ function generateEncryptionBlacklist(table: Table): string {
table.partitions
.flatMap((_) => _.models)
.forEach((model) => {
const [, pk] = model.keys.partitionKey
const [, sk] = model.keys.sortKey
encryptionBlacklistSet.add(pk.replace("$", ""))
encryptionBlacklistSet.add(sk.replace("$", ""))
const [, ...pkComponents] = model.keys.partitionKey
const [, ...skComponents] = model.keys.sortKey

pkComponents
.map(formatKeyComponent)
.forEach((_) => encryptionBlacklistSet.add(_))

skComponents
.map(formatKeyComponent)
.forEach((_) => encryptionBlacklistSet.add(_))
})

table.gsis.forEach(({ name, partitionKey, sortKey }) => {
encryptionBlacklistSet.add(partitionKey.replace("$", ""))
encryptionBlacklistSet.add(sortKey.replace("$", ""))
table.gsis.forEach(({ partitionKey, sortKey }) => {
encryptionBlacklistSet.add(formatKeyComponent(partitionKey))
encryptionBlacklistSet.add(formatKeyComponent(sortKey))
})

return JSON.stringify(Array.from(encryptionBlacklistSet))
return `[${Array.from(encryptionBlacklistSet).join(", ")}]`
}
2 changes: 1 addition & 1 deletion src/main/codegen/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ export type PartitionDefinition = {
}

export type ModelDefinition = Keys & Fields
export type Keys = { partitionKey: [string, string]; sortKey: [string, string] }
export type Keys = { partitionKey: string[]; sortKey: string[] }
export type Fields = { [fieldName: string]: string }

export type GSIDefinition = {
Expand Down
4 changes: 4 additions & 0 deletions src/main/codegen/util.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,7 @@ export function groupBy<T extends { [key: string]: any }, U extends keyof T>(

return Object.entries(groupedItems)
}

export function formatKeyComponent(component: string): string {
return `"${component.replace("$", "")}"`
}
52 changes: 29 additions & 23 deletions src/main/dynamo/Model.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,9 @@ export class Model<
constructor(
private table: Table,
private partitionKeyPrefix: string,
private partitionKeyField: U,
private partitionKeyFields: U[],
private sortKeyPrefix: string,
private sortKeyField: V,
private sortKeyFields: V[],
readonly modelTag: string
) {}

Expand All @@ -22,24 +22,27 @@ export class Model<
const {
partitionKeyPrefix,
sortKeyPrefix,
partitionKeyField,
sortKeyField,
partitionKeyFields,
sortKeyFields,
} = this
const pkComponents = partitionKeyFields.map((_) => params[_])
const skComponents = sortKeyFields.map((_) => params[_])
return new PartitionAndSortKey(
this.table.partitionKeyName,
this.buildKey(partitionKeyPrefix, params[partitionKeyField]),
this.buildKey(partitionKeyPrefix, ...pkComponents),

this.table.sortKeyName,
this.buildKey(sortKeyPrefix, params[sortKeyField]),
this.buildKey(sortKeyPrefix, ...skComponents),
this.modelTag
)
}

partitionKey(params: { [X in U]: string }): PartitionKeyAndSortKeyPrefix<T> {
const { partitionKeyPrefix, partitionKeyField } = this
const { partitionKeyPrefix, partitionKeyFields } = this
const pkComponents = partitionKeyFields.map((_) => params[_])
return new PartitionKeyAndSortKeyPrefix(
this.table.partitionKeyName,
this.buildKey(partitionKeyPrefix, params[partitionKeyField]),
this.buildKey(partitionKeyPrefix, ...pkComponents),
this.table.sortKeyName,
this.sortKeyPrefix,
this.modelTag
Expand All @@ -50,18 +53,18 @@ export class Model<
const {
partitionKeyPrefix,
sortKeyPrefix,
partitionKeyField,
sortKeyField,
partitionKeyFields,
sortKeyFields,
modelTag,
} = this

const fieldsWithTag = { ...fields, model: modelTag } as T

const pk = this.buildKey(
partitionKeyPrefix,
fieldsWithTag[partitionKeyField]
)
const sk = this.buildKey(sortKeyPrefix, fieldsWithTag[sortKeyField])
const pkComponents = partitionKeyFields.map((_) => fieldsWithTag[_])
const skComponents = sortKeyFields.map((_) => fieldsWithTag[_])

const pk = this.buildKey(partitionKeyPrefix, ...pkComponents)
const sk = this.buildKey(sortKeyPrefix, ...skComponents)

return {
...fieldsWithTag,
Expand All @@ -70,21 +73,21 @@ export class Model<
}
}

private buildKey(prefix: string, key: string): string {
return `${prefix}-${key}`
private buildKey(...components: string[]): string {
return components.join("-")
}
}

export class PartitionKeyBuilder<T extends TaggedModel> {
constructor(private table: Table, private modelTag: string) {}
partitionKey<U extends keyof T>(
prefix: string,
partitionKeyField: U
...partitionKeyFields: U[]
): SortKeyBuilder<T, U> {
return new SortKeyBuilder(
this.table,
prefix,
partitionKeyField,
partitionKeyFields,
this.modelTag
)
}
Expand All @@ -94,17 +97,20 @@ export class SortKeyBuilder<T extends TaggedModel, U extends keyof T> {
constructor(
private table: Table,
private partitionKeyPrefix: string,
private partitionKeyField: U,
private partitionKeyFields: U[],
private modelTag: string
) {}

sortKey<V extends keyof T>(prefix: string, sortKeyField: V): Model<T, U, V> {
sortKey<V extends keyof T>(
prefix: string,
...sortKeyFields: V[]
): Model<T, U, V> {
return new Model(
this.table,
this.partitionKeyPrefix,
this.partitionKeyField,
this.partitionKeyFields,
prefix,
sortKeyField,
sortKeyFields,
this.modelTag
)
}
Expand Down
48 changes: 47 additions & 1 deletion src/test/codegen/generateCode.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -176,7 +176,7 @@ export const MusiciansPartition = MusicTable.partition([MusicianModel])
`)
})

it("should generate table, add partition and sork key to encryption blacklist", () => {
it("should generate table, add partition and sort key to encryption blacklist", () => {
const result = generateCode(`
Tables:
Library:
Expand Down Expand Up @@ -322,3 +322,49 @@ Tables:

expect(lines).toContainEqual(`name: BestNameEvah`)
})

it("should generate a complex key model", () => {
const result = generateCode(`
Tables:
ComplexLibrary:
Partitions:
ComplexAuthors:
ComplexAuthor:
partitionKey: [Author, $id]
sortKey: [Author, $id, $name]
id: string
name: string
`)

expect(result).toEqual(`import { Table } from "@ginger.io/beyonce"

export const ComplexLibraryTable = new Table({
name: "ComplexLibrary",
partitionKeyName: "pk",
sortKeyName: "sk",
encryptionBlacklist: ["id", "name"]
})

export enum ModelType {
ComplexAuthor = "ComplexAuthor"
}

export interface ComplexAuthor {
model: ModelType.ComplexAuthor
id: string
name: string
}

export const ComplexAuthorModel = ComplexLibraryTable.model<ComplexAuthor>(
ModelType.ComplexAuthor
)
.partitionKey("Author", "id")
.sortKey("Author", "id", "name")

export type Model = ComplexAuthor

export const ComplexAuthorsPartition = ComplexLibraryTable.partition([
ComplexAuthorModel
])
`)
})
79 changes: 76 additions & 3 deletions src/test/dynamo/Beyonce.test.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import { FixedDataKeyProvider, JayZ } from "@ginger.io/jay-z"
import { DynamoDB } from "aws-sdk"
import crypto from "crypto"
import { Beyonce } from "../../main/dynamo/Beyonce"
import {
aMusicianWithTwoSongs,
byModelAndIdGSI,
Expand All @@ -9,18 +11,24 @@ import {
MusicianPartition,
Song,
SongModel,
table,
} from "./models"
import { setup } from "./util"
import { Beyonce } from "../../main/dynamo/Beyonce"
import { table } from "./models"
import { DynamoDB } from "aws-sdk"

describe("Beyonce", () => {
// Without encryption
it("should put and retrieve an item using pk + sk", async () => {
await testPutAndRetrieveItem()
})

it("should put and retrieve a model with a compound partition key", async () => {
await testPutAndRetrieveCompoundPartitionKey()
})

it("should put and retrieve a model with a compound sort key", async () => {
await testPutAndRetrieveCompoundSortKey()
})

it("should update a top-level item attribute", async () => {
const db = await setup()
const [musician, _, __] = aMusicianWithTwoSongs()
Expand Down Expand Up @@ -213,6 +221,16 @@ describe("Beyonce", () => {
await testPutAndRetrieveItem(jayZ)
})

it("should put and retrieve a model with a compound partition key with jayZ", async () => {
const jayZ = await createJayZ()
await testPutAndRetrieveCompoundPartitionKey(jayZ)
})

it("should put and retrieve a model with a compound sort key", async () => {
const jayZ = await createJayZ()
await testPutAndRetrieveCompoundSortKey(jayZ)
})

it("should put and delete an item using pk + sk with jayZ", async () => {
await testPutAndDeleteItem()
})
Expand Down Expand Up @@ -321,6 +339,61 @@ async function testEmptyQuery(jayZ?: JayZ) {
expect(result).toEqual({ musician: [], song: [] })
}

async function testPutAndRetrieveCompoundPartitionKey(jayZ?: JayZ) {
interface Person {
model: "person"
first: string
last: string
sortKey: string
}

const PersonModel = table
.model<Person>("person")
.partitionKey("Person", "first", "last")
.sortKey("Person", "sortKey")

const db = await setup(jayZ)
const model = PersonModel.create({
first: "Bob",
last: "Smith",
sortKey: "sortKey-123",
})
await db.put(model)

const result = await db.get(
PersonModel.key({ first: "Bob", last: "Smith", sortKey: "sortKey-123" })
)

expect(result).toEqual(model)
}

async function testPutAndRetrieveCompoundSortKey(jayZ?: JayZ) {
interface LineItem {
model: "lineItem"
orderId: string
id: string
timestamp: string
}

const LineItemModel = table
.model<LineItem>("lineItem")
.partitionKey("OrderId", "orderId")
.sortKey("LineItem", "id", "timestamp")

const db = await setup(jayZ)
const model = LineItemModel.create({
id: "l1",
orderId: "o1",
timestamp: "456",
})
await db.put(model)

const result = await db.get(
LineItemModel.key({ id: "l1", orderId: "o1", timestamp: "456" })
)
expect(result).toEqual(model)
}

async function testQueryWithPaginatedResults(jayZ?: JayZ) {
const db = await setup(jayZ)

Expand Down