@nmtjs/protocol 0.15.0-beta.3 → 0.15.0-beta.31

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (49) hide show
  1. package/dist/client/index.js +5 -5
  2. package/dist/client/protocol.js +1 -1
  3. package/dist/client/protocol.js.map +1 -1
  4. package/dist/client/stream.js +2 -2
  5. package/dist/client/stream.js.map +1 -1
  6. package/dist/client/versions/v1.d.ts +38 -38
  7. package/dist/client/versions/v1.js +3 -3
  8. package/dist/client/versions/v1.js.map +1 -1
  9. package/dist/common/binary.d.ts +3 -5
  10. package/dist/common/binary.js.map +1 -1
  11. package/dist/common/blob.d.ts +3 -3
  12. package/dist/common/blob.js +3 -3
  13. package/dist/common/blob.js.map +1 -1
  14. package/dist/common/enums.js +12 -6
  15. package/dist/common/enums.js.map +1 -1
  16. package/dist/common/index.js +6 -6
  17. package/dist/common/utils.js +1 -1
  18. package/dist/common/utils.js.map +1 -1
  19. package/dist/server/format.d.ts +5 -3
  20. package/dist/server/format.js +2 -0
  21. package/dist/server/format.js.map +1 -1
  22. package/dist/server/index.js +7 -7
  23. package/dist/server/protocol.js +1 -1
  24. package/dist/server/protocol.js.map +1 -1
  25. package/dist/server/stream.js.map +1 -1
  26. package/dist/server/utils.js.map +1 -1
  27. package/dist/server/versions/v1.d.ts +27 -27
  28. package/dist/server/versions/v1.js +3 -3
  29. package/dist/server/versions/v1.js.map +1 -1
  30. package/package.json +11 -13
  31. package/src/client/format.ts +49 -0
  32. package/src/client/index.ts +8 -0
  33. package/src/client/protocol.ts +107 -0
  34. package/src/client/stream.ts +222 -0
  35. package/src/client/versions/v1.ts +205 -0
  36. package/src/common/binary.ts +70 -0
  37. package/src/common/blob.ts +94 -0
  38. package/src/common/constants.ts +2 -0
  39. package/src/common/enums.ts +62 -0
  40. package/src/common/index.ts +6 -0
  41. package/src/common/types.ts +18 -0
  42. package/src/common/utils.ts +12 -0
  43. package/src/server/format.ts +117 -0
  44. package/src/server/index.ts +10 -0
  45. package/src/server/protocol.ts +97 -0
  46. package/src/server/stream.ts +51 -0
  47. package/src/server/types.ts +42 -0
  48. package/src/server/utils.ts +22 -0
  49. package/src/server/versions/v1.ts +198 -0
@@ -0,0 +1,51 @@
1
+ import type { ReadableOptions } from 'node:stream'
2
+ import { PassThrough, Readable } from 'node:stream'
3
+ import { ReadableStream } from 'node:stream/web'
4
+
5
+ import type { ProtocolBlob, ProtocolBlobMetadata } from '../common/blob.ts'
6
+
7
+ export class ProtocolClientStream extends PassThrough {
8
+ readonly #read?: ReadableOptions['read']
9
+
10
+ constructor(
11
+ public readonly id: number,
12
+ public readonly metadata: ProtocolBlobMetadata,
13
+ options?: ReadableOptions,
14
+ ) {
15
+ const { read, ...rest } = options ?? {}
16
+ super(rest)
17
+ this.#read = read
18
+ }
19
+
20
+ override _read(size: number): void {
21
+ if (this.#read) {
22
+ this.#read.call(this, size)
23
+ }
24
+ super._read(size)
25
+ }
26
+ }
27
+
28
+ export class ProtocolServerStream extends PassThrough {
29
+ public readonly id: number
30
+ public readonly metadata: ProtocolBlobMetadata
31
+
32
+ constructor(id: number, blob: ProtocolBlob) {
33
+ let readable: Readable
34
+
35
+ if (blob.source instanceof Readable) {
36
+ readable = blob.source
37
+ } else if (blob.source instanceof ReadableStream) {
38
+ readable = Readable.fromWeb(blob.source as ReadableStream)
39
+ } else {
40
+ throw new Error('Invalid source type')
41
+ }
42
+
43
+ super()
44
+
45
+ this.pause()
46
+ readable.pipe(this)
47
+
48
+ this.id = id
49
+ this.metadata = blob.metadata
50
+ }
51
+ }
@@ -0,0 +1,42 @@
1
+ import type { PlainType } from '@nmtjs/type'
2
+
3
+ import type {
4
+ ProtocolBlobInterface,
5
+ ProtocolBlobMetadata,
6
+ } from '../common/blob.ts'
7
+ import type { kBlobKey } from '../common/constants.ts'
8
+ import type { BaseServerDecoder, BaseServerEncoder } from './format.ts'
9
+ import type { ProtocolVersionInterface } from './protocol.ts'
10
+ import type { ProtocolClientStream } from './stream.ts'
11
+
12
+ export type ClientStreamConsumer = (() => ProtocolClientStream) & {
13
+ readonly [kBlobKey]: any
14
+ readonly metadata: ProtocolBlobMetadata
15
+ }
16
+
17
+ export type MessageContext = {
18
+ protocol: ProtocolVersionInterface
19
+ connectionId: string
20
+ streamId: () => number
21
+ decoder: BaseServerDecoder
22
+ encoder: BaseServerEncoder
23
+ addClientStream: (options: {
24
+ streamId: number
25
+ metadata: ProtocolBlobMetadata
26
+ callId: number
27
+ }) => ClientStreamConsumer
28
+ transport: {
29
+ send?: (connectionId: string, buffer: ArrayBufferView) => boolean | null
30
+ }
31
+ }
32
+
33
+ export type ResolveFormatParams = {
34
+ contentType?: string | null
35
+ accept?: string | null
36
+ }
37
+
38
+ export type InputType<T> = T extends ProtocolBlobInterface
39
+ ? ClientStreamConsumer
40
+ : T extends { [PlainType]?: true }
41
+ ? { [K in keyof Omit<T, PlainType>]: InputType<T[K]> }
42
+ : T
@@ -0,0 +1,22 @@
1
+ import type { ProtocolFormats } from './format.ts'
2
+ import type { ResolveFormatParams } from './types.ts'
3
+
4
+ export class UnsupportedFormatError extends Error {}
5
+
6
+ export class UnsupportedContentTypeError extends UnsupportedFormatError {}
7
+
8
+ export class UnsupportedAcceptTypeError extends UnsupportedFormatError {}
9
+
10
+ export const getFormat = (
11
+ format: ProtocolFormats,
12
+ { accept, contentType }: ResolveFormatParams,
13
+ ) => {
14
+ const encoder = contentType ? format.supportsEncoder(contentType) : undefined
15
+ if (!encoder)
16
+ throw new UnsupportedContentTypeError('Unsupported Content type')
17
+
18
+ const decoder = accept ? format.supportsDecoder(accept) : undefined
19
+ if (!decoder) throw new UnsupportedAcceptTypeError('Unsupported Accept type')
20
+
21
+ return { encoder, decoder }
22
+ }
@@ -0,0 +1,198 @@
1
+ import type { ServerMessageTypePayload } from '../protocol.ts'
2
+ import type { MessageContext } from '../types.ts'
3
+ import { decodeText, encodeNumber, encodeText } from '../../common/binary.ts'
4
+ import {
5
+ ClientMessageType,
6
+ MessageByteLength,
7
+ ProtocolVersion,
8
+ ServerMessageType,
9
+ } from '../../common/enums.ts'
10
+ import { ProtocolVersionInterface } from '../protocol.ts'
11
+
12
+ export class ProtocolVersion1 extends ProtocolVersionInterface {
13
+ version = ProtocolVersion.v1
14
+ decodeMessage(context: MessageContext, buffer: Buffer) {
15
+ const messageType = buffer.readUint8(0)
16
+ const messagePayload = buffer.subarray(MessageByteLength.MessageType)
17
+ switch (messageType) {
18
+ case ClientMessageType.Rpc: {
19
+ const callId = messagePayload.readUint32LE(0)
20
+ const procedureLength = messagePayload.readUInt16LE(
21
+ MessageByteLength.CallId,
22
+ )
23
+ const procedureOffset =
24
+ MessageByteLength.CallId + MessageByteLength.ProcedureLength
25
+ const procedure = messagePayload.toString(
26
+ 'utf-8',
27
+ procedureOffset,
28
+ procedureOffset + procedureLength,
29
+ )
30
+ const formatPayload = messagePayload.subarray(
31
+ procedureOffset + procedureLength,
32
+ )
33
+ const payload = context.decoder.decodeRPC(formatPayload, {
34
+ addStream: (streamId, metadata) => {
35
+ return context.addClientStream({ callId, streamId, metadata })
36
+ },
37
+ })
38
+
39
+ return { type: messageType, rpc: { callId, procedure, payload } }
40
+ }
41
+ case ClientMessageType.RpcPull: {
42
+ const callId = messagePayload.readUInt32LE(0)
43
+ return { type: messageType, callId }
44
+ }
45
+ case ClientMessageType.RpcAbort: {
46
+ const callId = messagePayload.readUInt32LE(0)
47
+ const reasonPayload = messagePayload.subarray(MessageByteLength.CallId)
48
+ const reason =
49
+ reasonPayload.byteLength > 0 ? decodeText(reasonPayload) : undefined
50
+ return { type: messageType, callId, reason }
51
+ }
52
+ case ClientMessageType.ServerStreamAbort: {
53
+ const streamId = messagePayload.readUInt32LE(0)
54
+ const reasonPayload = messagePayload.subarray(
55
+ MessageByteLength.StreamId,
56
+ )
57
+ const reason =
58
+ reasonPayload.byteLength > 0 ? decodeText(reasonPayload) : undefined
59
+ return { type: messageType, streamId, reason }
60
+ }
61
+ case ClientMessageType.ServerStreamPull: {
62
+ const streamId = messagePayload.readUInt32LE(0)
63
+ const size = messagePayload.readUInt32LE(MessageByteLength.StreamId)
64
+ return { type: messageType, streamId, size }
65
+ }
66
+ case ClientMessageType.ClientStreamAbort: {
67
+ const streamId = messagePayload.readUInt32LE(0)
68
+ const reasonPayload = messagePayload.subarray(
69
+ MessageByteLength.StreamId,
70
+ )
71
+ const reason =
72
+ reasonPayload.byteLength > 0 ? decodeText(reasonPayload) : undefined
73
+ return { type: messageType, streamId, reason }
74
+ }
75
+ case ClientMessageType.ClientStreamEnd: {
76
+ return { type: messageType, streamId: messagePayload.readUInt32LE(0) }
77
+ }
78
+ case ClientMessageType.ClientStreamPush: {
79
+ const streamId = messagePayload.readUInt32LE(0)
80
+ const chunk = messagePayload.subarray(MessageByteLength.StreamId)
81
+ return { type: messageType, streamId, chunk }
82
+ }
83
+
84
+ default:
85
+ throw new Error(`Unsupported message type: ${messageType}`)
86
+ }
87
+ }
88
+
89
+ encodeMessage<T extends ServerMessageType>(
90
+ context: MessageContext,
91
+ messageType: T,
92
+ payload: ServerMessageTypePayload[T],
93
+ ) {
94
+ switch (messageType) {
95
+ // case ServerMessageType.Event: {
96
+ // const { event, data } =
97
+ // payload as ServerMessageTypePayload[ServerMessageType.Event]
98
+ // return this.encode(
99
+ // encodeNumber(messageType, 'Uint8'),
100
+ // context.encoder.encode({ event, data }),
101
+ // )
102
+ // }
103
+ case ServerMessageType.RpcResponse: {
104
+ const { callId, result, streams, error } =
105
+ payload as ServerMessageTypePayload[ServerMessageType.RpcResponse]
106
+ return this.encode(
107
+ encodeNumber(messageType, 'Uint8'),
108
+ encodeNumber(callId, 'Uint32'),
109
+ encodeNumber(error ? 1 : 0, 'Uint8'),
110
+ error
111
+ ? context.encoder.encode(error)
112
+ : context.encoder.encodeRPC(result, streams),
113
+ )
114
+ }
115
+ case ServerMessageType.RpcStreamResponse: {
116
+ const { callId } =
117
+ payload as ServerMessageTypePayload[ServerMessageType.RpcStreamResponse]
118
+ return this.encode(
119
+ encodeNumber(messageType, 'Uint8'),
120
+ encodeNumber(callId, 'Uint32'),
121
+ )
122
+ }
123
+ case ServerMessageType.RpcStreamChunk: {
124
+ const { callId, chunk } =
125
+ payload as ServerMessageTypePayload[ServerMessageType.RpcStreamChunk]
126
+ return this.encode(
127
+ encodeNumber(messageType, 'Uint8'),
128
+ encodeNumber(callId, 'Uint32'),
129
+ chunk,
130
+ )
131
+ }
132
+ case ServerMessageType.RpcStreamEnd: {
133
+ const { callId } =
134
+ payload as ServerMessageTypePayload[ServerMessageType.RpcStreamEnd]
135
+ return this.encode(
136
+ encodeNumber(messageType, 'Uint8'),
137
+ encodeNumber(callId, 'Uint32'),
138
+ )
139
+ }
140
+ case ServerMessageType.RpcStreamAbort: {
141
+ const { callId, reason } =
142
+ payload as ServerMessageTypePayload[ServerMessageType.RpcStreamAbort]
143
+ return this.encode(
144
+ encodeNumber(messageType, 'Uint8'),
145
+ encodeNumber(callId, 'Uint32'),
146
+ reason ? encodeText(reason) : Buffer.alloc(0),
147
+ )
148
+ }
149
+ case ServerMessageType.ClientStreamPull: {
150
+ const { size, streamId } =
151
+ payload as ServerMessageTypePayload[ServerMessageType.ClientStreamPull]
152
+ return this.encode(
153
+ encodeNumber(messageType, 'Uint8'),
154
+ encodeNumber(streamId, 'Uint32'),
155
+ encodeNumber(size, 'Uint32'),
156
+ )
157
+ }
158
+ case ServerMessageType.ClientStreamAbort: {
159
+ const { streamId, reason } =
160
+ payload as ServerMessageTypePayload[ServerMessageType.ClientStreamAbort]
161
+ return this.encode(
162
+ encodeNumber(messageType, 'Uint8'),
163
+ encodeNumber(streamId, 'Uint32'),
164
+ reason ? encodeText(reason) : Buffer.alloc(0),
165
+ )
166
+ }
167
+ case ServerMessageType.ServerStreamPush: {
168
+ const { streamId, chunk } =
169
+ payload as ServerMessageTypePayload[ServerMessageType.ServerStreamPush]
170
+ return this.encode(
171
+ encodeNumber(messageType, 'Uint8'),
172
+ encodeNumber(streamId, 'Uint32'),
173
+ chunk,
174
+ )
175
+ }
176
+ case ServerMessageType.ServerStreamEnd: {
177
+ const { streamId } =
178
+ payload as ServerMessageTypePayload[ServerMessageType.ServerStreamEnd]
179
+ return this.encode(
180
+ encodeNumber(messageType, 'Uint8'),
181
+ encodeNumber(streamId, 'Uint32'),
182
+ )
183
+ }
184
+ case ServerMessageType.ServerStreamAbort: {
185
+ const { streamId, reason } =
186
+ payload as ServerMessageTypePayload[ServerMessageType.ServerStreamAbort]
187
+ return this.encode(
188
+ encodeNumber(messageType, 'Uint8'),
189
+ encodeNumber(streamId, 'Uint32'),
190
+ reason ? encodeText(reason) : Buffer.alloc(0),
191
+ )
192
+ }
193
+
194
+ default:
195
+ throw new Error(`Unsupported message type: ${messageType}`)
196
+ }
197
+ }
198
+ }