@xyo-network/archivist-cookie 2.85.6
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.
- package/LICENSE +165 -0
- package/README.md +13 -0
- package/dist/browser/CookieArchivist.d.cts +32 -0
- package/dist/browser/CookieArchivist.d.cts.map +1 -0
- package/dist/browser/CookieArchivist.d.mts +32 -0
- package/dist/browser/CookieArchivist.d.mts.map +1 -0
- package/dist/browser/CookieArchivist.d.ts +32 -0
- package/dist/browser/CookieArchivist.d.ts.map +1 -0
- package/dist/browser/index.cjs +156 -0
- package/dist/browser/index.cjs.map +1 -0
- package/dist/browser/index.d.cts +2 -0
- package/dist/browser/index.d.cts.map +1 -0
- package/dist/browser/index.d.mts +2 -0
- package/dist/browser/index.d.mts.map +1 -0
- package/dist/browser/index.d.ts +2 -0
- package/dist/browser/index.d.ts.map +1 -0
- package/dist/browser/index.js +125 -0
- package/dist/browser/index.js.map +1 -0
- package/dist/node/CookieArchivist.d.cts +32 -0
- package/dist/node/CookieArchivist.d.cts.map +1 -0
- package/dist/node/CookieArchivist.d.mts +32 -0
- package/dist/node/CookieArchivist.d.mts.map +1 -0
- package/dist/node/CookieArchivist.d.ts +32 -0
- package/dist/node/CookieArchivist.d.ts.map +1 -0
- package/dist/node/index.cjs +171 -0
- package/dist/node/index.cjs.map +1 -0
- package/dist/node/index.d.cts +2 -0
- package/dist/node/index.d.cts.map +1 -0
- package/dist/node/index.d.mts +2 -0
- package/dist/node/index.d.mts.map +1 -0
- package/dist/node/index.d.ts +2 -0
- package/dist/node/index.d.ts.map +1 -0
- package/dist/node/index.js +135 -0
- package/dist/node/index.js.map +1 -0
- package/package.json +75 -0
- package/src/CookieArchivist.ts +164 -0
- package/src/index.ts +1 -0
- package/src/spec/testArchivist.ts +50 -0
- package/typedoc.json +5 -0
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
import { assertEx } from '@xylabs/assert'
|
|
2
|
+
import { compact } from '@xylabs/lodash'
|
|
3
|
+
import { fulfilled, Promisable, PromisableArray } from '@xylabs/promise'
|
|
4
|
+
import { AbstractArchivist } from '@xyo-network/archivist-abstract'
|
|
5
|
+
import {
|
|
6
|
+
ArchivistAllQuerySchema,
|
|
7
|
+
ArchivistClearQuerySchema,
|
|
8
|
+
ArchivistCommitQuerySchema,
|
|
9
|
+
ArchivistConfig,
|
|
10
|
+
ArchivistDeleteQuerySchema,
|
|
11
|
+
ArchivistInsertQuery,
|
|
12
|
+
ArchivistInsertQuerySchema,
|
|
13
|
+
ArchivistModuleEventData,
|
|
14
|
+
ArchivistParams,
|
|
15
|
+
} from '@xyo-network/archivist-model'
|
|
16
|
+
import { BoundWitness } from '@xyo-network/boundwitness-model'
|
|
17
|
+
import { PayloadHasher } from '@xyo-network/hash'
|
|
18
|
+
import { AnyConfigSchema } from '@xyo-network/module-model'
|
|
19
|
+
import { Payload } from '@xyo-network/payload-model'
|
|
20
|
+
import { PayloadWrapper } from '@xyo-network/payload-wrapper'
|
|
21
|
+
import Cookies from 'js-cookie'
|
|
22
|
+
|
|
23
|
+
export type CookieArchivistConfigSchema = 'network.xyo.archivist.cookie.config'
|
|
24
|
+
export const CookieArchivistConfigSchema: CookieArchivistConfigSchema = 'network.xyo.archivist.cookie.config'
|
|
25
|
+
|
|
26
|
+
export type CookieArchivistConfig = ArchivistConfig<{
|
|
27
|
+
domain?: string
|
|
28
|
+
maxEntries?: number
|
|
29
|
+
maxEntrySize?: number
|
|
30
|
+
namespace?: string
|
|
31
|
+
schema: CookieArchivistConfigSchema
|
|
32
|
+
}>
|
|
33
|
+
|
|
34
|
+
export type CookieArchivistParams = ArchivistParams<AnyConfigSchema<CookieArchivistConfig>>
|
|
35
|
+
|
|
36
|
+
export class CookieArchivist<
|
|
37
|
+
TParams extends CookieArchivistParams,
|
|
38
|
+
TEventData extends ArchivistModuleEventData = ArchivistModuleEventData,
|
|
39
|
+
> extends AbstractArchivist<TParams, TEventData> {
|
|
40
|
+
static override configSchemas = [CookieArchivistConfigSchema]
|
|
41
|
+
|
|
42
|
+
get domain() {
|
|
43
|
+
return this.config?.domain
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
get maxEntries() {
|
|
47
|
+
//all browsers support at least 60 cookies
|
|
48
|
+
return this.config?.maxEntries ?? 60
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
get maxEntrySize() {
|
|
52
|
+
//all browsers support at least 4000 length per cookie
|
|
53
|
+
return this.config?.maxEntrySize ?? 4000
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
get namespace() {
|
|
57
|
+
return this.config?.namespace ?? 'xyoarch'
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
override get queries(): string[] {
|
|
61
|
+
return [
|
|
62
|
+
ArchivistAllQuerySchema,
|
|
63
|
+
ArchivistDeleteQuerySchema,
|
|
64
|
+
ArchivistClearQuerySchema,
|
|
65
|
+
ArchivistInsertQuerySchema,
|
|
66
|
+
ArchivistCommitQuerySchema,
|
|
67
|
+
...super.queries,
|
|
68
|
+
]
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
protected override allHandler(): PromisableArray<Payload> {
|
|
72
|
+
try {
|
|
73
|
+
return Object.entries(Cookies.get())
|
|
74
|
+
.filter(([key]) => key.startsWith(`${this.namespace}-`))
|
|
75
|
+
.map(([, value]) => JSON.parse(value))
|
|
76
|
+
} catch (ex) {
|
|
77
|
+
console.error(`Error: ${JSON.stringify(ex, null, 2)}`)
|
|
78
|
+
throw ex
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
protected override clearHandler(): void | Promise<void> {
|
|
83
|
+
try {
|
|
84
|
+
Object.entries(Cookies.get()).map(([key]) => {
|
|
85
|
+
if (key.startsWith(`${this.namespace}-`)) {
|
|
86
|
+
Cookies.remove(key)
|
|
87
|
+
}
|
|
88
|
+
})
|
|
89
|
+
} catch (ex) {
|
|
90
|
+
console.error(`Error: ${JSON.stringify(ex, null, 2)}`)
|
|
91
|
+
throw ex
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
protected override async commitHandler(): Promise<BoundWitness[]> {
|
|
96
|
+
try {
|
|
97
|
+
const payloads = await this.all()
|
|
98
|
+
assertEx(payloads.length > 0, 'Nothing to commit')
|
|
99
|
+
const settled = await Promise.allSettled(
|
|
100
|
+
compact(
|
|
101
|
+
Object.values((await this.parents()).commit ?? [])?.map(async (parent) => {
|
|
102
|
+
const queryPayload: ArchivistInsertQuery = {
|
|
103
|
+
schema: ArchivistInsertQuerySchema,
|
|
104
|
+
}
|
|
105
|
+
const query = await this.bindQuery(queryPayload, payloads)
|
|
106
|
+
return (await parent?.query(query[0], query[1]))?.[0]
|
|
107
|
+
}),
|
|
108
|
+
),
|
|
109
|
+
)
|
|
110
|
+
await this.clear()
|
|
111
|
+
return compact(settled.filter(fulfilled).map((result) => result.value))
|
|
112
|
+
} catch (ex) {
|
|
113
|
+
console.error(`Error: ${JSON.stringify(ex, null, 2)}`)
|
|
114
|
+
throw ex
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
protected override async deleteHandler(hashes: string[]): Promise<string[]> {
|
|
119
|
+
const payloadPairs: [string, Payload][] = await Promise.all(
|
|
120
|
+
(await this.get(hashes)).map<Promise<[string, Payload]>>(async (payload) => [await PayloadHasher.hashAsync(payload), payload]),
|
|
121
|
+
)
|
|
122
|
+
const deletedPairs: [string, Payload][] = compact(
|
|
123
|
+
await Promise.all(
|
|
124
|
+
payloadPairs.map<[string, Payload] | undefined>(([hash, payload]) => {
|
|
125
|
+
Cookies.remove(hash)
|
|
126
|
+
return [hash, payload]
|
|
127
|
+
}),
|
|
128
|
+
),
|
|
129
|
+
)
|
|
130
|
+
return deletedPairs.map(([hash]) => hash)
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
protected override getHandler(hashes: string[]): Promisable<Payload[]> {
|
|
134
|
+
return compact(
|
|
135
|
+
hashes.map((hash) => {
|
|
136
|
+
const cookieString = Cookies.get(this.keyFromHash(hash))
|
|
137
|
+
return cookieString ? JSON.parse(cookieString) : undefined
|
|
138
|
+
}),
|
|
139
|
+
)
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
protected override async insertHandler(payloads: Payload[]): Promise<Payload[]> {
|
|
143
|
+
try {
|
|
144
|
+
const resultPayloads: Payload[] = await Promise.all(
|
|
145
|
+
payloads.map(async (payload) => {
|
|
146
|
+
const wrapper = PayloadWrapper.wrap(payload)
|
|
147
|
+
const key = this.keyFromHash(await wrapper.hashAsync())
|
|
148
|
+
const value = JSON.stringify(wrapper.payload())
|
|
149
|
+
assertEx(value.length < this.maxEntrySize, `Payload too large [${wrapper.hashAsync()}, ${value.length}]`)
|
|
150
|
+
Cookies.set(key, JSON.stringify(wrapper.payload()))
|
|
151
|
+
return wrapper.payload()
|
|
152
|
+
}),
|
|
153
|
+
)
|
|
154
|
+
return resultPayloads
|
|
155
|
+
} catch (ex) {
|
|
156
|
+
console.error(`Error: ${JSON.stringify(ex, null, 2)}`)
|
|
157
|
+
throw ex
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
private keyFromHash(hash: string) {
|
|
162
|
+
return `${this.namespace}-${hash}`
|
|
163
|
+
}
|
|
164
|
+
}
|
package/src/index.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './CookieArchivist'
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @jest-environment jsdom
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { delay } from '@xylabs/delay'
|
|
6
|
+
import { Promisable } from '@xylabs/promise'
|
|
7
|
+
import { ArchivistInstance } from '@xyo-network/archivist-model'
|
|
8
|
+
import { IdSchema } from '@xyo-network/id-payload-plugin'
|
|
9
|
+
import { Payload } from '@xyo-network/payload-model'
|
|
10
|
+
import { PayloadWrapper } from '@xyo-network/payload-wrapper'
|
|
11
|
+
|
|
12
|
+
export const testArchivistRoundTrip = (archivistPromise: Promisable<ArchivistInstance>, name: string) => {
|
|
13
|
+
test(`Archivist RoundTrip [${name}]`, async () => {
|
|
14
|
+
const idPayload: Payload<{ salt: string }> = {
|
|
15
|
+
salt: Date.now().toString(),
|
|
16
|
+
schema: IdSchema,
|
|
17
|
+
}
|
|
18
|
+
const payloadWrapper = PayloadWrapper.wrap(idPayload)
|
|
19
|
+
|
|
20
|
+
const archivist = await archivistPromise
|
|
21
|
+
const insertResult = await archivist.insert([idPayload])
|
|
22
|
+
expect(insertResult).toBeDefined()
|
|
23
|
+
|
|
24
|
+
const getResult = await archivist.get([await payloadWrapper.hashAsync()])
|
|
25
|
+
expect(getResult).toBeDefined()
|
|
26
|
+
expect(getResult.length).toBe(1)
|
|
27
|
+
const gottenPayload = getResult[0]
|
|
28
|
+
if (gottenPayload) {
|
|
29
|
+
const gottenPayloadWrapper = PayloadWrapper.wrap(gottenPayload)
|
|
30
|
+
expect(await gottenPayloadWrapper.hashAsync()).toBe(await payloadWrapper.hashAsync())
|
|
31
|
+
}
|
|
32
|
+
})
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export const testArchivistAll = (archivist: Promisable<ArchivistInstance>, name: string) => {
|
|
36
|
+
test(`Archivist All [${name}]`, async () => {
|
|
37
|
+
const idPayload = {
|
|
38
|
+
salt: Date.now().toString(),
|
|
39
|
+
schema: IdSchema,
|
|
40
|
+
}
|
|
41
|
+
const archivistModule = await archivist
|
|
42
|
+
for (let x = 0; x < 10; x++) {
|
|
43
|
+
await archivistModule.insert([idPayload])
|
|
44
|
+
await delay(10)
|
|
45
|
+
}
|
|
46
|
+
const getResult = await archivistModule.all?.()
|
|
47
|
+
expect(getResult).toBeDefined()
|
|
48
|
+
expect(getResult?.length).toBe(2)
|
|
49
|
+
})
|
|
50
|
+
}
|