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

Help Creating Type-Safe Models #282

Open
kjrocker opened this issue Aug 14, 2023 · 4 comments
Open

Help Creating Type-Safe Models #282

kjrocker opened this issue Aug 14, 2023 · 4 comments

Comments

@kjrocker
Copy link

kjrocker commented Aug 14, 2023

I'm working on an application where all the BE types are compiled into Typescript for me. I would love to be able to enforce type-safety on the models in factory to some extent.

I've got the following code

type User = {
  id: string;
  firstName: string;
  lastName: string;
};

export const db = factory({
  user: {
    id: primaryKey(() => 'abc-123'),
    firstName: () => 'John',
    lastName: () => 'Maverick',
  }
})

Obviously because of the getters I can't just do user: { .... } as User

I also tried defining type Factory<T> = { [P in keyof T]-?: () => T[P] };

With that, I can do user: { .... } as Factory<User>, which works for simple fields, but breaks for anything using primaryKey, manyOf, oneOf, etc.

Any thoughts? I'd really just like to make sure I'm not missing keys when I define new models.

@kettanaito
Copy link
Member

Hi, @kjrocker. This is a valid concern. I'm working on the updated version of this library that covers improves type-safety as well. I can't promise to implement this into the existing version due to the lack of availability. But I do hope to get the new version out sometime soon (~this year).

In the next version, I tended to give the Schema generic to the consumer to specify the model on the type level:

type Models = {
  user: {
    id: Number
  }
  book: {
    id: number
    author: Relationship<'oneOf', 'user'>
  }
}

const db = factory<Models>({
  user: {
    id: id(Number)
  },
  book: {
    id: id(Number),
    author: oneOf('user')
  }
})

This is a bit tricky but I think I'm slowly getting it right internally. If you have some use cases to share please do, they can help a lot!

@pfe-nazaries
Copy link

Same problem here

@yslpn
Copy link

yslpn commented Nov 10, 2024

+++

@citrahtac
Copy link

citrahtac commented Nov 28, 2024

Try the below out. Props to ChatGpt though it took a painful amount of prompting. Sorry about the formatting. I can't get it straight.

import {
ModelDefinition,
ModelValueType,
PrimaryKeyType,
} from '@mswjs/data/lib/glossary';

import {
NullableProperty,
} from '@mswjs/data/lib/nullable';

import {
PrimaryKey,
} from '@mswjs/data/lib/primaryKey';

import {
ManyOf,
OneOf,
} from '@mswjs/data/lib/relations/Relation';

export type IFactory<
DTO_TYPE,
Relations extends { [K in keyof DTO_TYPE]?: string } = object,
PrimaryKeys extends keyof DTO_TYPE = Extract<'id', keyof DTO_TYPE>

= {
[K in keyof DTO_TYPE]: K extends PrimaryKeys // Check if the property is a primary key
? DTO_TYPE[K] extends PrimaryKeyType
? PrimaryKey<DTO_TYPE[K]>
: never
: K extends keyof Relations
? DTO_TYPE[K] extends Array
? ManyOf<NonNullable<Relations[K]>>
: OneOf<NonNullable<Relations[K]>>
: DTO_TYPE[K] extends ModelValueType
? () => DTO_TYPE[K]
: DTO_TYPE[K] extends ModelValueType | null
? NullableProperty<NonNullable<DTO_TYPE[K]>> | (() => DTO_TYPE[K])
: DTO_TYPE[K] extends object
? () => DTO_TYPE[K]
: never;
} & ModelDefinition;

export type IFactoryOptions<
DTO_TYPE,
Relations extends { [K in keyof DTO_TYPE]?: string } = object,
PrimaryKeys extends keyof DTO_TYPE = Extract<'id', keyof DTO_TYPE>

= IFactory<
DTO_TYPE,
Relations,
PrimaryKeys
;

USAGE:

export interface IMyComplexObjFactoryModel{
bigObjArrayFactoryModel: IBigObjModel[],
metaDataFactoryModel: IMetaDataFactoryModel,
somePrimaryId: number, // Unique identifier
}

export type IMyComplexObjFactoryOptions = IFactory<
IMyComplexObjFactoryModel,
{
bigObjArrayFactoryModel: 'bigObjArrayFactoryModel',
metaDataFactoryModel: 'metaDataFactoryModel'
},
'somePrimaryId'

const
myComplexObjFactoryModel:IMyComplexObjFactoryOptions = {
bigObjFactoryModel: manyOf('bigObjModel'),
somePrimaryId: primaryKey(() => faker.number.int()),
metaDataFactoryModel: oneOf('metaDataFactoryModel'),
};

factoryModelMock = <
T extends IFactoryOptions<
  object,
  object,
  never
>,
U

(
modelFactory: T,
): U => Object.assign(
Object.create(null),
factory({
bigObjModel: bigObjModel
metaDataFactoryModel: metaDataFactoryModel,
myComplexObjFactoryModel: myComplexObjFactoryModel
mock: modelFactory,
}).mock.create(),
);

myMock:IMyComplexObjFactoryModel = factoryModelMock<IMyComplexObjFactoryOptions , IMyComplexObjFactoryModel>(myComplexObjFactoryModel),

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

5 participants