• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

knowledgepixels / nanodash / 17911650819

22 Sep 2025 09:58AM UTC coverage: 13.581% (-0.03%) from 13.608%
17911650819

push

github

tkuhn
feat: Inverse and equivalent properties for Space member roles

435 of 4040 branches covered (10.77%)

Branch coverage included in aggregate %.

1115 of 7373 relevant lines covered (15.12%)

0.67 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

0.0
src/main/java/com/knowledgepixels/nanodash/Space.java
1
package com.knowledgepixels.nanodash;
2

3
import static com.knowledgepixels.nanodash.Utils.vf;
4

5
import java.io.Serializable;
6
import java.time.format.DateTimeParseException;
7
import java.util.ArrayList;
8
import java.util.Calendar;
9
import java.util.Collections;
10
import java.util.HashMap;
11
import java.util.HashSet;
12
import java.util.List;
13
import java.util.Map;
14
import java.util.Set;
15

16
import org.eclipse.rdf4j.model.IRI;
17
import org.eclipse.rdf4j.model.Literal;
18
import org.eclipse.rdf4j.model.Statement;
19
import org.eclipse.rdf4j.model.vocabulary.DCTERMS;
20
import org.nanopub.Nanopub;
21
import org.nanopub.extra.services.ApiResponse;
22
import org.nanopub.extra.services.ApiResponseEntry;
23
import org.nanopub.extra.services.QueryRef;
24
import org.nanopub.vocabulary.NTEMPLATE;
25
import org.slf4j.Logger;
26
import org.slf4j.LoggerFactory;
27

28
import com.github.jsonldjava.shaded.com.google.common.collect.Ordering;
29
import com.google.common.collect.ArrayListMultimap;
30
import com.google.common.collect.Multimap;
31
import com.knowledgepixels.nanodash.template.Template;
32
import com.knowledgepixels.nanodash.template.TemplateData;
33

34
import jakarta.xml.bind.DatatypeConverter;
35

36
/**
37
 * Class representing a "Space", which can be any kind of collaborative unit, like a project, group, or event.
38
 */
39
public class Space implements Serializable {
40

41
    private static final Logger logger = LoggerFactory.getLogger(Space.class);
×
42

43
    /**
44
     * The predicate to assign the admins of the space.
45
     */
46
    public static final IRI HAS_ADMIN = vf.createIRI("https://w3id.org/kpxl/gen/terms/hasAdmin");
×
47

48
    /**
49
     * The predicate for pinned templates in the space.
50
     */
51
    public static final IRI HAS_PINNED_TEMPLATE = vf.createIRI("https://w3id.org/kpxl/gen/terms/hasPinnedTemplate");
×
52

53
    /**
54
     * The predicate for pinned queries in the space.
55
     */
56
    public static final IRI HAS_PINNED_QUERY = vf.createIRI("https://w3id.org/kpxl/gen/terms/hasPinnedQuery");
×
57

58
    private static List<Space> spaceList;
59
    private static Map<String, List<Space>> spaceListByType;
60
    private static Map<String,Space> spacesByCoreInfo = new HashMap<>();
×
61
    private static Map<String,Space> spacesById;
62
    private static Map<Space,Set<Space>> subspaceMap;
63
    private static Map<Space,Set<Space>> superspaceMap;
64
    private static boolean loaded = false;
×
65

66
    public static synchronized void refresh(ApiResponse resp) {
67
        spaceList = new ArrayList<>();
×
68
        spaceListByType = new HashMap<>();
×
69
        Map<String,Space> prevSpacesByCoreInfoPrev = spacesByCoreInfo;
×
70
        spacesByCoreInfo = new HashMap<>();
×
71
        spacesById = new HashMap<>();
×
72
        subspaceMap = new HashMap<>();
×
73
        superspaceMap = new HashMap<>();
×
74
        for (ApiResponseEntry entry : resp.getData()) {
×
75
            Space space = new Space(entry);
×
76
            Space prevSpace = prevSpacesByCoreInfoPrev.get(space.getCoreInfoString());
×
77
            if (prevSpace != null) space = prevSpace;
×
78
            spaceList.add(space);
×
79
            spaceListByType.computeIfAbsent(space.getType(), k -> new ArrayList<>()).add(space);
×
80
            spacesByCoreInfo.put(space.getCoreInfoString(), space);
×
81
            spacesById.put(space.getId(), space);
×
82
        }
×
83
        for (Space space : spaceList) {
×
84
            Space superSpace = space.getIdSuperspace();
×
85
            if (superSpace == null) continue;
×
86
            subspaceMap.computeIfAbsent(superSpace, k -> new HashSet<>()).add(space);
×
87
            superspaceMap.computeIfAbsent(space, k -> new HashSet<>()).add(superSpace);
×
88
        }
×
89
        loaded = true;
×
90
    }
×
91

92
    public static boolean isLoaded() {
93
        return loaded;
×
94
    }
95

96
    public static void ensureLoaded() {
97
        if (spaceList == null) {
×
98
            refresh(QueryApiAccess.forcedGet(new QueryRef("get-spaces")));
×
99
        }
100
    }
×
101

102
    public static List<Space> getSpaceList() {
103
        ensureLoaded();
×
104
        return spaceList;
×
105
    }
106

107
    public static List<Space> getSpaceList(String type) {
108
        ensureLoaded();
×
109
        return spaceListByType.computeIfAbsent(type, k -> new ArrayList<>());
×
110
    }
111

112
    public static Space get(String id) {
113
        ensureLoaded();
×
114
        return spacesById.get(id);
×
115
    }
116

117
    public static void refresh() {
118
        ensureLoaded();
×
119
        for (Space space : spaceList) {
×
120
            space.dataNeedsUpdate = true;
×
121
        }
×
122
    }
×
123

124
    private String id, label, rootNanopubId, type;
125
    private Nanopub rootNanopub = null;
×
126
    private SpaceData data = new SpaceData();
×
127

128
    private static class SpaceData implements Serializable {
×
129

130
        String description = null;
×
131
        Calendar startDate, endDate;
132
        IRI defaultProvenance = null;
×
133

134
        List<IRI> admins = new ArrayList<>();
×
135
        Map<IRI,Set<SpaceMemberRole>> members = new HashMap<>();
×
136
        List<SpaceMemberRole> roles = new ArrayList<>();
×
137
        Map<IRI,SpaceMemberRole> roleMap = new HashMap<>();
×
138

139
        Map<String,IRI> adminPubkeyMap = new HashMap<>();
×
140
        List<Serializable> pinnedResources = new ArrayList<>();
×
141
        Set<String> pinGroupTags = new HashSet<>();
×
142
        Map<String, List<Serializable>> pinnedResourceMap = new HashMap<>();
×
143

144
        void addAdmin(IRI admin) {
145
            // TODO This isn't efficient for long owner lists:
146
            if (admins.contains(admin)) return;
×
147
            admins.add(admin);
×
148
            UserData ud = User.getUserData();
×
149
            for (String pubkeyhash : ud.getPubkeyhashes(admin, true)) {
×
150
                adminPubkeyMap.put(pubkeyhash, admin);
×
151
            }
×
152
        }
×
153

154
    }
155

156
    private boolean dataInitialized = false;
×
157
    private boolean dataNeedsUpdate = true;
×
158

159
    private Space(ApiResponseEntry resp) {
×
160
        this.id = resp.get("space");
×
161
        this.label = resp.get("label");
×
162
        this.type = resp.get("type");
×
163
        this.rootNanopubId = resp.get("np");
×
164
        this.rootNanopub = Utils.getAsNanopub(rootNanopubId);
×
165
        setCoreData(data);
×
166
    }
×
167

168
    public String getId() {
169
        return id;
×
170
    }
171

172
    public String getRootNanopubId() {
173
        return rootNanopubId;
×
174
    }
175

176
    public String getCoreInfoString() {
177
        return id + " " + rootNanopubId;
×
178
    }
179

180
    public Nanopub getRootNanopub() {
181
        return rootNanopub;
×
182
    }
183

184
    public String getLabel() {
185
        return label;
×
186
    }
187

188
    public String getType() {
189
        return type;
×
190
    }
191

192
    public Calendar getStartDate() {
193
        return data.startDate;
×
194
    }
195

196
    public Calendar getEndDate() {
197
        return data.endDate;
×
198
    }
199

200
    public String getTypeLabel() {
201
        return type.replaceFirst("^.*/", "");
×
202
    }
203

204
    public String getDescription() {
205
        return data.description;
×
206
    }
207

208
    public boolean isDataInitialized() {
209
        triggerDataUpdate();
×
210
        return dataInitialized;
×
211
    }
212

213
    public List<IRI> getAdmins() {
214
        triggerDataUpdate();
×
215
        return data.admins;
×
216
    }
217

218
    public List<IRI> getMembers() {
219
        triggerDataUpdate();
×
220
        List<IRI> members = new ArrayList<IRI>(data.members.keySet());
×
221
        members.sort(User.getUserData().userComparator);
×
222
        return members;
×
223
    }
224

225
    public Set<SpaceMemberRole> getMemberRoles(IRI memberId) {
226
        return data.members.get(memberId);
×
227
    }
228

229
    public boolean isMember(IRI userId) {
230
        triggerDataUpdate();
×
231
        return data.members.containsKey(userId);
×
232
    }
233

234
    public List<Serializable> getPinnedResources() {
235
        triggerDataUpdate();
×
236
        return data.pinnedResources;
×
237
    }
238

239
    public Set<String> getPinGroupTags() {
240
        triggerDataUpdate();
×
241
        return data.pinGroupTags;
×
242
    }
243

244
    public Map<String, List<Serializable>> getPinnedResourceMap() {
245
        triggerDataUpdate();
×
246
        return data.pinnedResourceMap;
×
247
    }
248

249
    public IRI getDefaultProvenance() {
250
        return data.defaultProvenance;
×
251
    }
252

253
    public List<SpaceMemberRole> getRoles() {
254
        return data.roles;
×
255
    }
256

257
    public String getSuperId() {
258
        return null;
×
259
    }
260

261
    public Space getIdSuperspace() {
262
        if (!id.matches("https?://[^/]+/.*/[^/]*/?")) return null;
×
263
        String superId = id.replaceFirst("(https?://[^/]+/.*)/[^/]*/?", "$1");
×
264
        if (spacesById.containsKey(superId)) {
×
265
            return spacesById.get(superId);
×
266
        }
267
        return null;
×
268
    }
269

270
    public List<Space> getSuperspaces() {
271
        if (superspaceMap.containsKey(this)) {
×
272
            List<Space> superspaces = new ArrayList<>(superspaceMap.get(this));
×
273
            Collections.sort(superspaces, Ordering.usingToString());
×
274
            return superspaces;
×
275
        }
276
        return new ArrayList<>();
×
277
    }
278

279
    public List<Space> getSubspaces() {
280
        if (subspaceMap.containsKey(this)) {
×
281
            List<Space> subspaces = new ArrayList<>(subspaceMap.get(this));
×
282
            Collections.sort(subspaces, Ordering.usingToString());
×
283
            return subspaces;
×
284
        }
285
        return new ArrayList<>();
×
286
    }
287

288
    public List<Space> getSubspaces(String type) {
289
        List<Space> l = new ArrayList<>();
×
290
        for (Space s : getSubspaces()) {
×
291
            if (s.getType().equals(type)) l.add(s);
×
292
        }
×
293
        return l;
×
294
    }
295

296
    private synchronized void triggerDataUpdate() {
297
        if (dataNeedsUpdate) {
×
298
            new Thread(() -> {
×
299
                SpaceData newData = new SpaceData();
×
300
                setCoreData(newData);
×
301

302
                newData.roles.add(SpaceMemberRole.ADMIN_ROLE);
×
303
                newData.roleMap.put(SpaceMemberRole.ADMIN_ROLE_IRI, SpaceMemberRole.ADMIN_ROLE);
×
304

305
                for (ApiResponseEntry r : QueryApiAccess.forcedGet(new QueryRef("get-admins", "unit", id)).getData()) {
×
306
                    String pubkeyhash = r.get("pubkeyhash");
×
307
                    if (newData.adminPubkeyMap.containsKey(pubkeyhash)) {
×
308
                        IRI adminId = Utils.vf.createIRI(r.get("admin"));
×
309
                        newData.addAdmin(adminId);
×
310
                        newData.members.computeIfAbsent(adminId, (k) -> new HashSet<>()).add(SpaceMemberRole.ADMIN_ROLE);
×
311
                    }
312
                }
×
313
                newData.admins.sort(User.getUserData().userComparator);
×
314

315
                Multimap<String, String> getSpaceMemberParams = ArrayListMultimap.create();
×
316
                getSpaceMemberParams.put("space", id);
×
317

318
                for (ApiResponseEntry r : QueryApiAccess.forcedGet(new QueryRef( "get-space-member-roles", "space", id)).getData()) {
×
319
                    if (!newData.adminPubkeyMap.containsKey(r.get("pubkey"))) continue;
×
320
                    SpaceMemberRole role = new SpaceMemberRole(r);
×
321
                    newData.roles.add(role);
×
322

323
                    // TODO Handle cases of overlapping properties:
324
                    newData.roleMap.put(role.getMainProperty(), role);
×
325
                    for (IRI p : role.getEquivalentProperties()) newData.roleMap.put(p, role);
×
326
                    for (IRI p : role.getInverseProperties()) newData.roleMap.put(p, role);
×
327
    
328
                    role.addRoleParams(getSpaceMemberParams);
×
329
                }
×
330

331
                for (ApiResponseEntry r : QueryApiAccess.forcedGet(new QueryRef("get-space-members", getSpaceMemberParams)).getData()) {
×
332
                    IRI memberId = Utils.vf.createIRI(r.get("member"));
×
333
                    SpaceMemberRole role = newData.roleMap.get(Utils.vf.createIRI(r.get("role")));
×
334
                    newData.members.computeIfAbsent(memberId, (k) -> new HashSet<>()).add(role);
×
335
                }
×
336

337
                for (ApiResponseEntry r : QueryApiAccess.forcedGet(new QueryRef("get-pinned-templates", "space", id)).getData()) {
×
338
                    if (!newData.adminPubkeyMap.containsKey(r.get("pubkey"))) continue;
×
339
                    Template t = TemplateData.get().getTemplate(r.get("template"));
×
340
                    if (t == null) continue;
×
341
                    newData.pinnedResources.add(t);
×
342
                    String tag = r.get("tag");
×
343
                    if (tag != null && !tag.isEmpty()) {
×
344
                        newData.pinGroupTags.add(r.get("tag"));
×
345
                        newData.pinnedResourceMap.computeIfAbsent(tag, k -> new ArrayList<>()).add(TemplateData.get().getTemplate(r.get("template")));
×
346
                    }
347
                }
×
348
                for (ApiResponseEntry r : QueryApiAccess.forcedGet(new QueryRef("get-pinned-queries", "space", id)).getData()) {
×
349
                    if (!newData.adminPubkeyMap.containsKey(r.get("pubkey"))) continue;
×
350
                    GrlcQuery query = GrlcQuery.get(r.get("query"));
×
351
                    if (query == null) continue;
×
352
                    newData.pinnedResources.add(query);
×
353
                    String tag = r.get("tag");
×
354
                    if (tag != null && !tag.isEmpty()) {
×
355
                        newData.pinGroupTags.add(r.get("tag"));
×
356
                        newData.pinnedResourceMap.computeIfAbsent(tag, k -> new ArrayList<>()).add(query);
×
357
                    }
358
                }
×
359
                data = newData;
×
360
                dataInitialized = true;
×
361
            }).start();
×
362
            dataNeedsUpdate = false;
×
363
        }
364
    }
×
365

366
    private void setCoreData(SpaceData data) {
367
        for (Statement st : rootNanopub.getAssertion()) {
×
368
            if (st.getSubject().stringValue().equals(getId())) {
×
369
                if (st.getPredicate().equals(DCTERMS.DESCRIPTION)) {
×
370
                    data.description = st.getObject().stringValue();
×
371
                } else if (st.getPredicate().stringValue().equals("http://schema.org/startDate")) {
×
372
                    try {
373
                        data.startDate = DatatypeConverter.parseDateTime(st.getObject().stringValue());
×
374
                    } catch (DateTimeParseException ex) {
×
375
                        logger.error("Failed to parse date {}", st.getObject().stringValue());
×
376
                    }
×
377
                } else if (st.getPredicate().stringValue().equals("http://schema.org/endDate")) {
×
378
                    try {
379
                        data.endDate = DatatypeConverter.parseDateTime(st.getObject().stringValue());
×
380
                    } catch (IllegalArgumentException ex) {
×
381
                        logger.error("Failed to parse date {}", st.getObject().stringValue());
×
382
                    }
×
383
                } else if (st.getPredicate().equals(HAS_ADMIN) && st.getObject() instanceof IRI obj) {
×
384
                    data.addAdmin(obj);
×
385
                } else if (st.getPredicate().equals(HAS_PINNED_TEMPLATE) && st.getObject() instanceof IRI obj) {
×
386
                    data.pinnedResources.add(TemplateData.get().getTemplate(obj.stringValue()));
×
387
                } else if (st.getPredicate().equals(HAS_PINNED_QUERY) && st.getObject() instanceof IRI obj) {
×
388
                    data.pinnedResources.add(GrlcQuery.get(obj.stringValue()));
×
389
                } else if (st.getPredicate().equals(NTEMPLATE.HAS_DEFAULT_PROVENANCE) && st.getObject() instanceof IRI obj) {
×
390
                    data.defaultProvenance = obj;
×
391
                }
392
            } else if (st.getPredicate().equals(NTEMPLATE.HAS_TAG) && st.getObject() instanceof Literal l) {
×
393
                data.pinGroupTags.add(l.stringValue());
×
394
                List<Serializable> list = data.pinnedResourceMap.get(l.stringValue());
×
395
                if (list == null) {
×
396
                    list = new ArrayList<>();
×
397
                    data.pinnedResourceMap.put(l.stringValue(), list);
×
398
                }
399
                list.add(TemplateData.get().getTemplate(st.getSubject().stringValue()));
×
400
            }
401
        }
×
402
    }
×
403

404
    @Override
405
    public String toString() {
406
        return id;
×
407
    }
408

409
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc