mirror of
https://github.com/zenstackhq/zenstack
synced 2026-05-24 10:08:55 +00:00
* fix(policy): run pg/sqlite tests, misc dual db compatibility fixes * addressing review comments, cleaning up text search casing * addressing pr comments * try fixing CI * update * fix tests * update
244 lines
5.9 KiB
TypeScript
244 lines
5.9 KiB
TypeScript
import { sql } from 'kysely';
|
|
import { afterEach, describe, expect, it } from 'vitest';
|
|
import { createTestClient } from '../utils';
|
|
|
|
describe('Computed fields tests', () => {
|
|
let db: any;
|
|
|
|
afterEach(async () => {
|
|
await db?.$disconnect();
|
|
});
|
|
|
|
it('works with non-optional fields', async () => {
|
|
db = await createTestClient(
|
|
`
|
|
model User {
|
|
id Int @id @default(autoincrement())
|
|
name String
|
|
upperName String @computed
|
|
}
|
|
`,
|
|
{
|
|
computedFields: {
|
|
User: {
|
|
upperName: (eb: any) => eb.fn('upper', ['name']),
|
|
},
|
|
},
|
|
} as any,
|
|
);
|
|
|
|
await expect(
|
|
db.user.create({
|
|
data: { id: 1, name: 'Alex' },
|
|
}),
|
|
).resolves.toMatchObject({
|
|
upperName: 'ALEX',
|
|
});
|
|
|
|
await expect(
|
|
db.user.findUnique({
|
|
where: { id: 1 },
|
|
select: { upperName: true },
|
|
}),
|
|
).resolves.toMatchObject({
|
|
upperName: 'ALEX',
|
|
});
|
|
|
|
await expect(
|
|
db.user.findFirst({
|
|
where: { upperName: 'ALEX' },
|
|
}),
|
|
).resolves.toMatchObject({
|
|
upperName: 'ALEX',
|
|
});
|
|
|
|
await expect(
|
|
db.user.findFirst({
|
|
where: { upperName: 'Alex' },
|
|
}),
|
|
).toResolveNull();
|
|
|
|
await expect(
|
|
db.user.findFirst({
|
|
orderBy: { upperName: 'desc' },
|
|
}),
|
|
).resolves.toMatchObject({
|
|
upperName: 'ALEX',
|
|
});
|
|
|
|
await expect(
|
|
db.user.findFirst({
|
|
orderBy: { upperName: 'desc' },
|
|
take: 1,
|
|
}),
|
|
).resolves.toMatchObject({
|
|
upperName: 'ALEX',
|
|
});
|
|
|
|
await expect(
|
|
db.user.aggregate({
|
|
_count: { upperName: true },
|
|
}),
|
|
).resolves.toMatchObject({
|
|
_count: { upperName: 1 },
|
|
});
|
|
|
|
await expect(
|
|
db.user.groupBy({
|
|
by: ['upperName'],
|
|
_count: { upperName: true },
|
|
_max: { upperName: true },
|
|
}),
|
|
).resolves.toEqual([
|
|
expect.objectContaining({
|
|
_count: { upperName: 1 },
|
|
_max: { upperName: 'ALEX' },
|
|
}),
|
|
]);
|
|
});
|
|
|
|
it('is typed correctly for non-optional fields', async () => {
|
|
db = await createTestClient(
|
|
`
|
|
model User {
|
|
id Int @id @default(autoincrement())
|
|
name String
|
|
upperName String @computed
|
|
}
|
|
`,
|
|
{
|
|
extraSourceFiles: {
|
|
main: `
|
|
import { ZenStackClient } from '@zenstackhq/runtime';
|
|
import { schema } from './schema';
|
|
|
|
async function main() {
|
|
const client = new ZenStackClient(schema, {
|
|
dialect: {} as any,
|
|
computedFields: {
|
|
User: {
|
|
upperName: (eb) => eb.fn('upper', ['name']),
|
|
},
|
|
}
|
|
});
|
|
|
|
const user = await client.user.create({
|
|
data: { id: 1, name: 'Alex' }
|
|
});
|
|
console.log(user.upperName);
|
|
// @ts-expect-error
|
|
user.upperName = null;
|
|
}
|
|
|
|
main();
|
|
`,
|
|
},
|
|
},
|
|
);
|
|
});
|
|
|
|
it('works with optional fields', async () => {
|
|
db = await createTestClient(
|
|
`
|
|
model User {
|
|
id Int @id @default(autoincrement())
|
|
name String
|
|
upperName String? @computed
|
|
}
|
|
`,
|
|
{
|
|
computedFields: {
|
|
User: {
|
|
upperName: (eb: any) => eb.lit(null),
|
|
},
|
|
},
|
|
} as any,
|
|
);
|
|
|
|
await expect(
|
|
db.user.create({
|
|
data: { id: 1, name: 'Alex' },
|
|
}),
|
|
).resolves.toMatchObject({
|
|
upperName: null,
|
|
});
|
|
});
|
|
|
|
it('is typed correctly for optional fields', async () => {
|
|
db = await createTestClient(
|
|
`
|
|
model User {
|
|
id Int @id @default(autoincrement())
|
|
name String
|
|
upperName String? @computed
|
|
}
|
|
`,
|
|
{
|
|
extraSourceFiles: {
|
|
main: `
|
|
import { ZenStackClient } from '@zenstackhq/runtime';
|
|
import { schema } from './schema';
|
|
|
|
async function main() {
|
|
const client = new ZenStackClient(schema, {
|
|
dialect: {} as any,
|
|
computedFields: {
|
|
User: {
|
|
upperName: (eb) => eb.lit(null),
|
|
},
|
|
}
|
|
});
|
|
|
|
const user = await client.user.create({
|
|
data: { id: 1, name: 'Alex' }
|
|
});
|
|
console.log(user.upperName);
|
|
user.upperName = null;
|
|
}
|
|
|
|
main();
|
|
`,
|
|
},
|
|
},
|
|
);
|
|
});
|
|
|
|
it('works with read from a relation', async () => {
|
|
db = await createTestClient(
|
|
`
|
|
model User {
|
|
id Int @id @default(autoincrement())
|
|
name String
|
|
posts Post[]
|
|
postCount Int @computed
|
|
}
|
|
|
|
model Post {
|
|
id Int @id @default(autoincrement())
|
|
title String
|
|
author User @relation(fields: [authorId], references: [id])
|
|
authorId Int
|
|
}
|
|
`,
|
|
{
|
|
computedFields: {
|
|
User: {
|
|
postCount: (eb: any, context: { modelAlias: string }) =>
|
|
eb
|
|
.selectFrom('Post')
|
|
.whereRef('Post.authorId', '=', sql.ref(`${context.modelAlias}.id`))
|
|
.select(() => eb.fn.countAll().as('count')),
|
|
},
|
|
},
|
|
} as any,
|
|
);
|
|
|
|
await db.user.create({
|
|
data: { id: 1, name: 'Alex', posts: { create: { title: 'Post1' } } },
|
|
});
|
|
|
|
await expect(db.post.findFirst({ select: { id: true, author: true } })).resolves.toMatchObject({
|
|
author: expect.objectContaining({ postCount: 1 }),
|
|
});
|
|
});
|
|
});
|