forked from drizzle-team/drizzle-orm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsession.ts
142 lines (113 loc) · 4.58 KB
/
session.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
import { entityKind } from '~/entity';
import { TransactionRollbackError } from '~/errors';
import { type RelationalSchemaConfig, type TablesRelationalConfig } from '~/relations';
import { type Query, type SQL, sql } from '~/sql';
import { type Assume, type Equal } from '~/utils';
import { MySqlDatabase } from './db';
import type { MySqlDialect } from './dialect';
import type { SelectedFieldsOrdered } from './query-builders/select.types';
export type Mode = 'default' | 'planetscale';
export interface QueryResultHKT {
readonly $brand: 'MySqlQueryRowHKT';
readonly row: unknown;
readonly type: unknown;
}
export type QueryResultKind<TKind extends QueryResultHKT, TRow> = (TKind & {
readonly row: TRow;
})['type'];
export interface PreparedQueryConfig {
execute: unknown;
iterator: unknown;
}
export interface PreparedQueryHKT {
readonly $brand: 'MySqlPreparedQueryHKT';
readonly config: unknown;
readonly type: unknown;
}
export type PreparedQueryKind<
TKind extends PreparedQueryHKT,
TConfig extends PreparedQueryConfig,
TAssume extends boolean = false,
> = Equal<TAssume, true> extends true ? Assume<(TKind & { readonly config: TConfig })['type'], PreparedQuery<TConfig>>
: (TKind & { readonly config: TConfig })['type'];
export abstract class PreparedQuery<T extends PreparedQueryConfig> {
static readonly [entityKind]: string = 'MySqlPreparedQuery';
/** @internal */
joinsNotNullableMap?: Record<string, boolean>;
abstract execute(placeholderValues?: Record<string, unknown>): Promise<T['execute']>;
abstract iterator(placeholderValues?: Record<string, unknown>): AsyncGenerator<T['iterator']>;
}
export interface MySqlTransactionConfig {
withConsistentSnapshot?: boolean;
accessMode?: 'read only' | 'read write';
isolationLevel: 'read uncommitted' | 'read committed' | 'repeatable read' | 'serializable';
}
export abstract class MySqlSession<
TQueryResult extends QueryResultHKT = QueryResultHKT,
TPreparedQueryHKT extends PreparedQueryHKTBase = PreparedQueryHKTBase,
TFullSchema extends Record<string, unknown> = Record<string, never>,
TSchema extends TablesRelationalConfig = Record<string, never>,
> {
static readonly [entityKind]: string = 'MySqlSession';
constructor(protected dialect: MySqlDialect) {}
abstract prepareQuery<T extends PreparedQueryConfig, TPreparedQueryHKT extends PreparedQueryHKT>(
query: Query,
fields: SelectedFieldsOrdered | undefined,
customResultMapper?: (rows: unknown[][]) => T['execute'],
): PreparedQueryKind<TPreparedQueryHKT, T>;
execute<T>(query: SQL): Promise<T> {
return this.prepareQuery<PreparedQueryConfig & { execute: T }, PreparedQueryHKTBase>(
this.dialect.sqlToQuery(query),
undefined,
).execute();
}
abstract all<T = unknown>(query: SQL): Promise<T[]>;
abstract transaction<T>(
transaction: (tx: MySqlTransaction<TQueryResult, TPreparedQueryHKT, TFullSchema, TSchema>) => Promise<T>,
config?: MySqlTransactionConfig,
): Promise<T>;
protected getSetTransactionSQL(config: MySqlTransactionConfig): SQL | undefined {
const parts: string[] = [];
if (config.isolationLevel) {
parts.push(`isolation level ${config.isolationLevel}`);
}
return parts.length ? sql.join(['set transaction ', parts.join(' ')]) : undefined;
}
protected getStartTransactionSQL(config: MySqlTransactionConfig): SQL | undefined {
const parts: string[] = [];
if (config.withConsistentSnapshot) {
parts.push('with consistent snapshot');
}
if (config.accessMode) {
parts.push(config.accessMode);
}
return parts.length ? sql.join(['start transaction ', parts.join(' ')]) : undefined;
}
}
export abstract class MySqlTransaction<
TQueryResult extends QueryResultHKT,
TPreparedQueryHKT extends PreparedQueryHKTBase,
TFullSchema extends Record<string, unknown> = Record<string, never>,
TSchema extends TablesRelationalConfig = Record<string, never>,
> extends MySqlDatabase<TQueryResult, TPreparedQueryHKT, TFullSchema, TSchema> {
static readonly [entityKind]: string = 'MySqlTransaction';
constructor(
dialect: MySqlDialect,
session: MySqlSession,
protected schema: RelationalSchemaConfig<TSchema> | undefined,
protected readonly nestedIndex: number,
mode: Mode,
) {
super(dialect, session, schema, mode);
}
rollback(): never {
throw new TransactionRollbackError();
}
/** Nested transactions (aka savepoints) only work with InnoDB engine. */
abstract override transaction<T>(
transaction: (tx: MySqlTransaction<TQueryResult, TPreparedQueryHKT, TFullSchema, TSchema>) => Promise<T>,
): Promise<T>;
}
export interface PreparedQueryHKTBase extends PreparedQueryHKT {
type: PreparedQuery<Assume<this['config'], PreparedQueryConfig>>;
}