@exodus/atoms 5.2.1 → 5.2.3
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/package.json +4 -4
- package/src/enhancers/combine.js +33 -11
- package/src/enhancers/dedupe.js +7 -1
- package/src/factories/storage.js +12 -0
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@exodus/atoms",
|
|
3
|
-
"version": "5.2.
|
|
3
|
+
"version": "5.2.3",
|
|
4
4
|
"main": "src/index.js",
|
|
5
5
|
"description": "Abstraction for encapsulating a piece of data behind a simple unified interface: get, set, observe",
|
|
6
6
|
"author": "Exodus Movement Inc.",
|
|
@@ -16,7 +16,7 @@
|
|
|
16
16
|
"type": "git",
|
|
17
17
|
"url": "git+https://github.com/ExodusMovement/exodus-hydra.git"
|
|
18
18
|
},
|
|
19
|
-
"homepage": "https://github.com/ExodusMovement/exodus-hydra/tree/master/
|
|
19
|
+
"homepage": "https://github.com/ExodusMovement/exodus-hydra/tree/master/libraries/atoms",
|
|
20
20
|
"license": "UNLICENSED",
|
|
21
21
|
"bugs": {
|
|
22
22
|
"url": "https://github.com/ExodusMovement/exodus-hydra/issues?q=is%3Aissue+is%3Aopen+label%3Aatom"
|
|
@@ -32,8 +32,8 @@
|
|
|
32
32
|
"devDependencies": {
|
|
33
33
|
"@exodus/storage-memory": "^2.1.0",
|
|
34
34
|
"delay": "^5.0.0",
|
|
35
|
-
"eslint": "^8.
|
|
35
|
+
"eslint": "^8.44.0",
|
|
36
36
|
"jest": "^29.1.2"
|
|
37
37
|
},
|
|
38
|
-
"gitHead": "
|
|
38
|
+
"gitHead": "ced5da1161e6f9b688ad4571cdb9c9a6b07adb8b"
|
|
39
39
|
}
|
package/src/enhancers/combine.js
CHANGED
|
@@ -3,24 +3,46 @@ import createSimpleObserver from '../simple-observer'
|
|
|
3
3
|
import enforceObservableRules from '../enforce-rules'
|
|
4
4
|
|
|
5
5
|
const combine = (atoms) => {
|
|
6
|
-
const { notify, observe } = createSimpleObserver()
|
|
6
|
+
const { notify, observe: observeSimpleObserver } = createSimpleObserver()
|
|
7
7
|
const countdownLock = createCountdownLock(Object.keys(atoms))
|
|
8
8
|
|
|
9
9
|
let values = {}
|
|
10
|
+
let subscriptions = []
|
|
11
|
+
let subscribers = 0
|
|
10
12
|
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
13
|
+
const maybeUnsubscribeSourceObservables = () => {
|
|
14
|
+
subscribers -= 1
|
|
15
|
+
if (subscribers > 0) return
|
|
16
|
+
subscriptions.forEach((unsubscribe) => unsubscribe())
|
|
17
|
+
subscriptions = []
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const observe = (callback) => {
|
|
21
|
+
subscribers += 1
|
|
22
|
+
|
|
23
|
+
if (subscribers === 1) {
|
|
24
|
+
subscriptions = Object.entries(atoms).map(([name, atom]) =>
|
|
25
|
+
atom.observe(async (value) => {
|
|
26
|
+
values = {
|
|
27
|
+
...values,
|
|
28
|
+
[name]: value,
|
|
29
|
+
}
|
|
30
|
+
if (countdownLock.unlock(name)) await notify(values)
|
|
31
|
+
})
|
|
32
|
+
)
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
const unsubscribe = observeSimpleObserver(callback)
|
|
36
|
+
return () => {
|
|
37
|
+
unsubscribe()
|
|
38
|
+
maybeUnsubscribeSourceObservables()
|
|
39
|
+
}
|
|
19
40
|
}
|
|
20
41
|
|
|
21
42
|
const get = async () => {
|
|
22
|
-
|
|
23
|
-
|
|
43
|
+
return Object.fromEntries(
|
|
44
|
+
await Promise.all(Object.entries(atoms).map(async ([name, atom]) => [name, await atom.get()]))
|
|
45
|
+
)
|
|
24
46
|
}
|
|
25
47
|
|
|
26
48
|
const set = async () => {
|
package/src/enhancers/dedupe.js
CHANGED
|
@@ -1,7 +1,13 @@
|
|
|
1
1
|
import { isEqual } from 'lodash'
|
|
2
2
|
|
|
3
3
|
const dedupe = (atom) => {
|
|
4
|
-
const set = (value) =>
|
|
4
|
+
const set = (value) => {
|
|
5
|
+
// avoid triggering a get() when we're resetting the atom
|
|
6
|
+
// TODO: add atom.unset and stop using atom.set(undefined)
|
|
7
|
+
if (value === undefined) return atom.set(value)
|
|
8
|
+
|
|
9
|
+
return atom.set((previous) => (isEqual(previous, value) ? previous : value))
|
|
10
|
+
}
|
|
5
11
|
|
|
6
12
|
return {
|
|
7
13
|
...atom,
|
package/src/factories/storage.js
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import createSimpleObserver from '../simple-observer'
|
|
2
2
|
import enforceObservableRules from '../enforce-rules'
|
|
3
|
+
import pDefer from 'p-defer'
|
|
3
4
|
|
|
4
5
|
const createStorageAtomFactory =
|
|
5
6
|
({ storage, logger }) =>
|
|
@@ -7,12 +8,16 @@ const createStorageAtomFactory =
|
|
|
7
8
|
const { notify, observe } = createSimpleObserver({ enable: isSoleWriter })
|
|
8
9
|
|
|
9
10
|
let cached
|
|
11
|
+
let writePromiseDefer
|
|
10
12
|
const set = async (value) => {
|
|
11
13
|
if (value === undefined) {
|
|
14
|
+
writePromiseDefer = pDefer()
|
|
12
15
|
await storage.delete(key)
|
|
13
16
|
} else {
|
|
17
|
+
writePromiseDefer = pDefer()
|
|
14
18
|
await storage.set(key, value)
|
|
15
19
|
}
|
|
20
|
+
writePromiseDefer.resolve()
|
|
16
21
|
|
|
17
22
|
if (isSoleWriter) {
|
|
18
23
|
cached = value
|
|
@@ -25,6 +30,13 @@ const createStorageAtomFactory =
|
|
|
25
30
|
return cached
|
|
26
31
|
}
|
|
27
32
|
|
|
33
|
+
if (writePromiseDefer) {
|
|
34
|
+
await writePromiseDefer.promise
|
|
35
|
+
}
|
|
36
|
+
if (cached) {
|
|
37
|
+
return cached
|
|
38
|
+
}
|
|
39
|
+
|
|
28
40
|
const value = await storage.get(key)
|
|
29
41
|
|
|
30
42
|
if (isSoleWriter) {
|