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
|
// Copyright (C) 2023 Vaughn Nugent
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
import { Endpoints } from "./server-api";
import { NostrPubKey, Watchable } from "./types";
import {
type FeatureApi,
type IFeatureExport,
type BgRuntime,
optionsOnly,
popupAndOptionsOnly,
exportForegroundApi
} from "./framework";
import { AppSettings } from "./settings";
import { shallowRef, watch } from "vue";
import { useSession } from "@vnuge/vnlib.browser";
import { set, useToggle } from "@vueuse/core";
import { defer, isArray } from "lodash";
import { waitForChange, waitForChangeFn } from "./util";
export interface IdentityApi extends FeatureApi, Watchable {
createIdentity: (identity: NostrPubKey) => Promise<NostrPubKey>
updateIdentity: (identity: NostrPubKey) => Promise<NostrPubKey>
deleteIdentity: (key: NostrPubKey) => Promise<void>
getAllKeys: () => Promise<NostrPubKey[]>;
getPublicKey: () => Promise<NostrPubKey | undefined>;
selectKey: (key: NostrPubKey) => Promise<void>;
}
export const useIdentityApi = (): IFeatureExport<AppSettings, IdentityApi> => {
return{
background: ({ state }: BgRuntime<AppSettings>) =>{
const { execRequest } = state.useServerApi();
const { loggedIn } = useSession();
//Get the current selected key
const selectedKey = shallowRef<NostrPubKey | undefined>();
const allKeys = shallowRef<NostrPubKey[]>([]);
const [ onKeyUpdateTriggered , triggerKeyUpdate ] = useToggle()
const keyLoadWatchLoop = async () => {
while(true){
//Load keys from server if logged in
if(loggedIn.value){
const [...keys] = await execRequest(Endpoints.GetKeys);
allKeys.value = isArray(keys) ? keys : [];
}
else{
//Clear all keys when logged out
allKeys.value = [];
}
//Wait for changes to trigger a new key-load
await waitForChange([loggedIn, onKeyUpdateTriggered])
}
}
defer(keyLoadWatchLoop)
//Clear the selected key if the user logs out
watch(loggedIn, (li) => li ? null : selectedKey.value = undefined)
return {
//Identity is only available in options context
createIdentity: optionsOnly(async (id: NostrPubKey) => {
const newKey = await execRequest(Endpoints.CreateId, id)
triggerKeyUpdate()
return newKey
}),
updateIdentity: optionsOnly(async (id: NostrPubKey) => {
const updated = await execRequest(Endpoints.UpdateId, id)
triggerKeyUpdate()
return updated
}),
deleteIdentity: optionsOnly(async (key: NostrPubKey) => {
await execRequest(Endpoints.DeleteKey, key);
triggerKeyUpdate()
}),
selectKey: popupAndOptionsOnly((key: NostrPubKey): Promise<void> => {
set(selectedKey, key);
return Promise.resolve()
}),
getAllKeys: (): Promise<NostrPubKey[]> => {
return Promise.resolve(allKeys.value);
},
getPublicKey: (): Promise<NostrPubKey | undefined> => {
return Promise.resolve(selectedKey.value);
},
waitForChange: waitForChangeFn([selectedKey, loggedIn, allKeys])
}
},
foreground: exportForegroundApi([
'createIdentity',
'updateIdentity',
'deleteIdentity',
'getAllKeys',
'getPublicKey',
'selectKey',
'waitForChange'
])
}
}
|