-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathcontactLogic.js
232 lines (210 loc) · 7.78 KB
/
contactLogic.js
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
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
// Logic for solid contacts
import * as UI from 'solid-ui'
import { store } from 'solid-logic'
import { getPersonas } from './webidControl'
const ns = UI.ns
const $rdf = UI.rdf
const utils = UI.utils
const kb = store
const updater = kb.updater
/** Perform updates on more than one document @@ Move to rdflib!
*/
export async function updateMany (deletions, insertions = []) {
const docs = deletions.concat(insertions).map(st => st.why)
const uniqueDocs = []
docs.forEach(doc => {
if (!uniqueDocs.find(uniqueDoc => uniqueDoc.equals(doc))) uniqueDocs.push(doc)
})
const updates = uniqueDocs.map(doc =>
kb.updater.update(deletions.filter(st => st.why.sameTerm(doc)),
insertions.filter(st => st.why.sameTerm(doc))))
return Promise.all(updates)
}
/** Add a new person to the web data
*
* adds them to the given groups as well.
* @returns {NamedNode} the person
*/
export async function saveNewContact (book, name, selectedGroups, klass) {
await kb.fetcher.load(book.doc())
const nameEmailIndex = kb.any(book, ns.vcard('nameEmailIndex'))
const uuid = utils.genUuid()
const person = kb.sym(
book.dir().uri + 'Person/' + uuid + '/index.ttl#this'
)
const doc = person.doc()
// Set of statements to different files
const agenda = [
// Patch the main index to add the person
$rdf.st(person, ns.vcard('inAddressBook'), book, nameEmailIndex), // The people index
$rdf.st(person, ns.vcard('fn'), name, nameEmailIndex),
// The new person file
$rdf.st(person, ns.vcard('fn'), name, doc),
$rdf.st(person, ns.rdf('type'), klass, doc),
$rdf.st(doc, ns.dct('created'), new Date(), doc) // Note when created - useful for triaging later
// Note this is propert of the file -- not when the person was created!
]
for (const gu in selectedGroups) {
const g = kb.sym(gu)
const gd = g.doc()
agenda.push(
$rdf.st(g, ns.vcard('hasMember'), person, gd),
$rdf.st(person, ns.vcard('fn'), name, gd)
)
}
try {
await updater.updateMany([], agenda) // @@ in future, updater.updateMany
} catch (e) {
console.error("Error: can't update " + person + ' as new contact:' + e)
throw new Error('Updating new contact: ' + e)
}
return person
}
export function sanitizeToAlpha (name) { // https://mathiasbynens.be/notes/es6-unicode-regex
const n2 = name.replace(/\W/gu, '_') // Anything which is not a unicode word characeter
return n2.replace(/_+/g, '_') // https://www.regular-expressions.info/shorthand.html
}
/** Write new group to web
* Creates an empty new group file and adds it to the index
* @returns group
*/
export async function saveNewGroup (book, name) {
await kb.fetcher.load(book.doc())
const gix = kb.any(book, ns.vcard('groupIndex'))
const gname = sanitizeToAlpha(name)
const group = kb.sym(book.dir().uri + 'Group/' + gname + '.ttl#this')
const doc = group.doc()
// console.log(' New group will be: ' + group + '\n')
try {
await kb.fetcher.load(gix)
} catch (err) {
throw new Error('Error loading group index!' + gix.uri + ': ' + err)
}
if (kb.holds(book, ns.vcard('includesGroup'), group, gix)) {
return group // Already exists
}
const insertTriples = [
$rdf.st(book, ns.vcard('includesGroup'), group, gix),
$rdf.st(group, ns.rdf('type'), ns.vcard('Group'), gix),
$rdf.st(group, ns.vcard('fn'), name, gix)
]
try {
await updater.update([], insertTriples)
} catch (e) {
throw new Error('Could not update group index ' + e) // fail
}
const triples = [
$rdf.st(book, ns.vcard('includesGroup'), group, doc), // Pointer back to book
$rdf.st(group, ns.rdf('type'), ns.vcard('Group'), doc),
$rdf.st(group, ns.vcard('fn'), name, doc)
]
try {
await updater.update([], triples)
} catch (err) {
throw new Error('Could not update group file: ' + err) // fail
}
return group
}
export async function addPersonToGroup (thing, group) {
const toBeFetched = [thing.doc(), group.doc()]
try {
await kb.fetcher.load(toBeFetched)
} catch (e) {
throw new Error('addPersonToGroup: ' + e)
}
const types = kb.findTypeURIs(thing)
for (const ty in types) {
// console.log(' drop object type includes: ' + ty) // @@ Allow email addresses and phone numbers to be dropped?
}
if (!(ns.vcard('Individual').uri in types ||
ns.vcard('Organization').uri in types)) {
return alert(`Can't add ${thing} to a group: it has to be an individual or another group.`)
}
const pname = kb.any(thing, ns.vcard('fn'))
const gname = kb.any(group, ns.vcard('fn'))
if (!pname) { return alert('No vcard name known for ' + thing) }
const already = kb.holds(thing, ns.vcard('fn'), null, group.doc())
if (already) {
return alert(
'ALREADY added ' + pname + ' to group ' + gname
)
}
const message = 'Add ' + pname + ' to group ' + gname + '?'
if (!confirm(message)) return
const ins = [
$rdf.st(thing, ns.vcard('fn'), pname, group.doc())
]
// find person webIDs and insert in vcard:hasMember
const webIDs = getPersonas(kb, thing).map(webid => webid.value)
if (webIDs.length) {
webIDs.forEach(webid => {
ins.push($rdf.st(kb.sym(webid), ns.owl('sameAs'), thing, group.doc()))
ins.push($rdf.st(group, ns.vcard('hasMember'), kb.sym(webid), group.doc()))
})
} else {
ins.push($rdf.st(group, ns.vcard('hasMember'), thing, group.doc()))
}
try {
await updater.update([], ins)
// to allow refresh of card groupList
kb.fetcher.unload(group.doc())
await kb.fetcher.load(group.doc())
} catch (e) {
throw new Error(`Error adding ${pname} to group ${gname}:` + e)
}
return thing
}
/**
* Find persons member of a group
*/
export function groupMembers (kb, group) {
const a = kb.each(group, ns.vcard('hasMember'), null, group.doc())
let b = []
a.forEach(item => {
/* const contacts = kb.each(item, ns.owl('sameAs'), null, group.doc())
if (contacts.length) {
if (!kb.any(contacts[0], ns.vard('fn'))) b = b.concat(item) // this is the old data model
else b = b.concat(contacts)
} else { b = b.concat(item) }
b = b.concat(item) */
// to keep compatibility with old data model
// check if item is a contact, else it is a WebID and parse 'sameAs' for contacts
b = kb.any(item, ns.vcard('fn'), null, group.doc()) ? b.concat(item) : b.concat(kb.each(item, ns.owl('sameAs'), null, group.doc()))
})
const strings = new Set(b.map(contact => contact.uri)) // remove dups
b = [...strings].map(uri => kb.sym(uri))
return b
}
export function isLocal(group, item) {
const tree = group.dir().dir().dir()
const local = item.uri && item.uri.startsWith(tree.uri)
// console.log(` isLocal ${local} for ${item.uri} in group ${group} tree ${tree.uri}`)
return local
}
export function getSameAs(kb, item, doc) {
return kb.each(item, ns.owl('sameAs'), null, doc).concat(
kb.each(null, ns.owl('sameAs'), item, doc))
}
export async function getDataModelIssues(groups) {
let del = []
let ins = []
groups.forEach(group => {
const members = kb.each(group, ns.vcard('hasMember'), null, group.doc())
members.forEach((member) => {
const others = getSameAs(kb, member, group.doc())
if (others.length && isLocal(group, member)) { // Problem: local ID used instead of webID
for (const other of others) {
if (!isLocal(group, other)) { // Let's use this one as the immediate member for CSS ACLs'
// console.warn(`getDataModelIssues: Need to swap ${member} to ${other}`)
del.push($rdf.st(group, ns.vcard('hasMember'), member, group.doc()))
ins.push($rdf.st(group, ns.vcard('hasMember'), other, group.doc()))
break
}
// console.log('getDataModelIssues: ??? expected id not to be local ' + other)
} // other
} // if
}) // member
}) // next group
return {del, ins }
} // getDataModelIssues
// Ends