1
- import { Client , CreateOptions , RedisConnection , RedisHashData , RedisJsonData } from '../client'
2
- import { Entity , EntityId , EntityKeyName } from '../entity'
3
- import { buildRediSearchSchema } from '../indexer'
4
- import { Schema } from '../schema'
5
- import { Search , RawSearch } from '../search'
6
- import { fromRedisHash , fromRedisJson , toRedisHash , toRedisJson } from '../transformer'
1
+ import { Client , CreateOptions , RedisConnection , RedisHashData , RedisJsonData } from '../client'
2
+ import { Entity , EntityId , EntityKeyName } from '../entity'
3
+ import { buildRediSearchSchema } from '../indexer'
4
+ import { Schema } from '../schema'
5
+ import { RawSearch , Search } from '../search'
6
+ import { fromRedisHash , fromRedisJson , toRedisHash , toRedisJson } from '../transformer'
7
7
8
8
/**
9
9
* A repository is the main interaction point for reading, writing, and
@@ -41,19 +41,19 @@ import { fromRedisHash, fromRedisJson, toRedisHash, toRedisJson } from '../trans
41
41
* .and('aBoolean').is.false().returnAll()
42
42
* ```
43
43
*/
44
- export class Repository {
44
+ export class Repository < T extends Entity = Record < string , any > > {
45
45
46
46
// NOTE: Not using "#" private as the spec needs to check calls on this class. Will be resolved when Client class is removed.
47
- private client : Client
48
- #schema: Schema
47
+ private readonly client : Client
48
+ readonly #schema: Schema < T >
49
49
50
50
/**
51
51
* Creates a new {@link Repository}.
52
52
*
53
53
* @param schema The schema defining that data in the repository.
54
- * @param client A client to talk to Redis.
54
+ * @param clientOrConnection A client to talk to Redis.
55
55
*/
56
- constructor ( schema : Schema , clientOrConnection : Client | RedisConnection ) {
56
+ constructor ( schema : Schema < T > , clientOrConnection : Client | RedisConnection ) {
57
57
this . #schema = schema
58
58
if ( clientOrConnection instanceof Client ) {
59
59
this . client = clientOrConnection
@@ -131,7 +131,7 @@ export class Repository {
131
131
* @param entity The Entity to save.
132
132
* @returns A copy of the provided Entity with EntityId and EntityKeyName properties added.
133
133
*/
134
- async save ( entity : Entity ) : Promise < Entity >
134
+ async save ( entity : T ) : Promise < T >
135
135
136
136
/**
137
137
* Insert or update the {@link Entity} to Redis using the provided entityId.
@@ -140,10 +140,10 @@ export class Repository {
140
140
* @param entity The Entity to save.
141
141
* @returns A copy of the provided Entity with EntityId and EntityKeyName properties added.
142
142
*/
143
- async save ( id : string , entity : Entity ) : Promise < Entity >
143
+ async save ( id : string , entity : T ) : Promise < T >
144
144
145
- async save ( entityOrId : Entity | string , maybeEntity ?: Entity ) : Promise < Entity > {
146
- let entity : Entity | undefined
145
+ async save ( entityOrId : T | string , maybeEntity ?: T ) : Promise < T > {
146
+ let entity : T | undefined
147
147
let entityId : string | undefined
148
148
149
149
if ( typeof entityOrId !== 'string' ) {
@@ -155,7 +155,7 @@ export class Repository {
155
155
}
156
156
157
157
const keyName = `${ this . #schema. schemaName } :${ entityId } `
158
- const clonedEntity = { ...entity , [ EntityId ] : entityId , [ EntityKeyName ] : keyName }
158
+ const clonedEntity = { ...entity , [ EntityId ] : entityId , [ EntityKeyName ] : keyName } as T
159
159
await this . writeEntity ( clonedEntity )
160
160
161
161
return clonedEntity
@@ -168,7 +168,7 @@ export class Repository {
168
168
* @param id The ID of the {@link Entity} you seek.
169
169
* @returns The matching Entity.
170
170
*/
171
- async fetch ( id : string ) : Promise < Entity >
171
+ async fetch ( id : string ) : Promise < T >
172
172
173
173
/**
174
174
* Read and return the {@link Entity | Entities} from Redis with the given IDs. If
@@ -177,7 +177,7 @@ export class Repository {
177
177
* @param ids The IDs of the {@link Entity | Entities} you seek.
178
178
* @returns The matching Entities.
179
179
*/
180
- async fetch ( ...ids : string [ ] ) : Promise < Entity [ ] >
180
+ async fetch ( ...ids : string [ ] ) : Promise < T [ ] >
181
181
182
182
/**
183
183
* Read and return the {@link Entity | Entities} from Redis with the given IDs. If
@@ -186,9 +186,9 @@ export class Repository {
186
186
* @param ids The IDs of the {@link Entity | Entities} you seek.
187
187
* @returns The matching Entities.
188
188
*/
189
- async fetch ( ids : string [ ] ) : Promise < Entity [ ] >
189
+ async fetch ( ids : string [ ] ) : Promise < T [ ] >
190
190
191
- async fetch ( ids : string | string [ ] ) : Promise < Entity | Entity [ ] > {
191
+ async fetch ( ids : string | string [ ] ) : Promise < T | T [ ] > {
192
192
if ( arguments . length > 1 ) return this . readEntities ( [ ...arguments ] )
193
193
if ( Array . isArray ( ids ) ) return this . readEntities ( ids )
194
194
@@ -246,6 +246,7 @@ export class Repository {
246
246
* ids. If a particular {@link Entity} is not found, does nothing.
247
247
*
248
248
* @param ids The IDs of the {@link Entity | Entities} you wish to delete.
249
+ * @param ttlInSeconds The time to live in seconds.
249
250
*/
250
251
async expire ( ids : string [ ] , ttlInSeconds : number ) : Promise < void >
251
252
@@ -298,7 +299,7 @@ export class Repository {
298
299
*
299
300
* @returns A {@link Search} object.
300
301
*/
301
- search ( ) : Search {
302
+ search ( ) : Search < T > {
302
303
return new Search ( this . #schema, this . client )
303
304
}
304
305
@@ -313,20 +314,19 @@ export class Repository {
313
314
* @query The raw RediSearch query you want to rune.
314
315
* @returns A {@link RawSearch} object.
315
316
*/
316
- searchRaw ( query : string ) : RawSearch {
317
+ searchRaw ( query : string ) : RawSearch < T > {
317
318
return new RawSearch ( this . #schema, this . client , query )
318
319
}
319
320
320
- private async writeEntity ( entity : Entity ) : Promise < void > {
321
- return this . #schema. dataStructure === 'HASH' ? this . writeEntityToHash ( entity ) : this . writeEntityToJson ( entity )
321
+ private async writeEntity ( entity : T ) : Promise < void > {
322
+ return this . #schema. dataStructure === 'HASH' ? this . # writeEntityToHash( entity ) : this . writeEntityToJson ( entity )
322
323
}
323
324
324
- private async readEntities ( ids : string [ ] ) : Promise < Entity [ ] > {
325
+ private async readEntities ( ids : string [ ] ) : Promise < T [ ] > {
325
326
return this . #schema. dataStructure === 'HASH' ? this . readEntitiesFromHash ( ids ) : this . readEntitiesFromJson ( ids )
326
327
}
327
328
328
- // TODO: make this actually private... like with #
329
- private async writeEntityToHash ( entity : Entity ) : Promise < void > {
329
+ async #writeEntityToHash( entity : Entity ) : Promise < void > {
330
330
const keyName = entity [ EntityKeyName ] !
331
331
const hashData : RedisHashData = toRedisHash ( this . #schema, entity )
332
332
if ( Object . keys ( hashData ) . length === 0 ) {
@@ -336,14 +336,13 @@ export class Repository {
336
336
}
337
337
}
338
338
339
- private async readEntitiesFromHash ( ids : string [ ] ) : Promise < Entity [ ] > {
339
+ private async readEntitiesFromHash ( ids : string [ ] ) : Promise < T [ ] > {
340
340
return Promise . all (
341
- ids . map ( async ( entityId ) => {
341
+ ids . map ( async ( entityId ) : Promise < T > => {
342
342
const keyName = this . makeKey ( entityId )
343
343
const hashData = await this . client . hgetall ( keyName )
344
344
const entityData = fromRedisHash ( this . #schema, hashData )
345
- const entity = { ...entityData , [ EntityId ] : entityId , [ EntityKeyName ] : keyName }
346
- return entity
345
+ return { ...entityData , [ EntityId ] : entityId , [ EntityKeyName ] : keyName } as T
347
346
} ) )
348
347
}
349
348
@@ -353,14 +352,13 @@ export class Repository {
353
352
await this . client . jsonset ( keyName , jsonData )
354
353
}
355
354
356
- private async readEntitiesFromJson ( ids : string [ ] ) : Promise < Entity [ ] > {
355
+ private async readEntitiesFromJson ( ids : string [ ] ) : Promise < T [ ] > {
357
356
return Promise . all (
358
- ids . map ( async ( entityId ) => {
357
+ ids . map ( async ( entityId ) : Promise < T > => {
359
358
const keyName = this . makeKey ( entityId )
360
359
const jsonData = await this . client . jsonget ( keyName ) ?? { }
361
360
const entityData = fromRedisJson ( this . #schema, jsonData )
362
- const entity = { ...entityData , [ EntityId ] : entityId , [ EntityKeyName ] : keyName }
363
- return entity
361
+ return { ...entityData , [ EntityId ] : entityId , [ EntityKeyName ] : keyName } as T
364
362
} ) )
365
363
}
366
364
0 commit comments