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

knowledgepixels / nanodash / 25678638590

11 May 2026 03:08PM UTC coverage: 20.512% (+2.2%) from 18.296%
25678638590

push

github

web-flow
Merge pull request #455 from knowledgepixels/fix-repetition-resets-edited-values

fix: don't reseed shared model from URL param on repetition (#271)

1013 of 6236 branches covered (16.24%)

Branch coverage included in aggregate %.

2592 of 11339 relevant lines covered (22.86%)

3.28 hits per line

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

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

3
import com.knowledgepixels.nanodash.LocalUri;
4
import com.knowledgepixels.nanodash.RestrictedChoice;
5
import com.knowledgepixels.nanodash.domain.User;
6
import com.knowledgepixels.nanodash.Utils;
7
import com.knowledgepixels.nanodash.component.StatementItem.RepetitionGroup;
8
import com.knowledgepixels.nanodash.domain.IndividualAgent;
9
import com.knowledgepixels.nanodash.domain.MaintainedResource;
10
import com.knowledgepixels.nanodash.domain.Space;
11
import com.knowledgepixels.nanodash.page.ExplorePage;
12
import com.knowledgepixels.nanodash.repository.MaintainedResourceRepository;
13
import com.knowledgepixels.nanodash.repository.SpaceRepository;
14
import com.knowledgepixels.nanodash.template.ContextType;
15
import com.knowledgepixels.nanodash.template.Template;
16
import com.knowledgepixels.nanodash.template.UnificationException;
17
import net.trustyuri.TrustyUriUtils;
18
import org.apache.commons.codec.Charsets;
19
import org.apache.wicket.behavior.AttributeAppender;
20
import org.apache.wicket.markup.html.basic.Label;
21
import org.apache.wicket.markup.html.link.ExternalLink;
22
import org.apache.wicket.model.IModel;
23
import org.apache.wicket.model.Model;
24
import org.apache.wicket.validation.IValidatable;
25
import org.apache.wicket.validation.IValidator;
26
import org.apache.wicket.validation.Validatable;
27
import org.apache.wicket.validation.ValidationError;
28
import org.eclipse.rdf4j.common.net.ParsedIRI;
29
import org.eclipse.rdf4j.model.IRI;
30
import org.eclipse.rdf4j.model.Literal;
31
import org.eclipse.rdf4j.model.Value;
32
import org.eclipse.rdf4j.model.ValueFactory;
33
import org.eclipse.rdf4j.model.impl.SimpleValueFactory;
34
import org.eclipse.rdf4j.model.util.Literals;
35
import org.eclipse.rdf4j.model.vocabulary.XSD;
36
import org.nanopub.Nanopub;
37
import org.nanopub.NanopubUtils;
38
import org.nanopub.SimpleCreatorPattern;
39
import org.nanopub.vocabulary.NTEMPLATE;
40
import org.slf4j.Logger;
41
import org.slf4j.LoggerFactory;
42

43
import java.net.URISyntaxException;
44
import java.net.URLEncoder;
45
import java.util.HashMap;
46
import java.util.Map;
47

48
/**
49
 * ReadonlyItem is a component that displays a read-only item in the form.
50
 */
51
public class ReadonlyItem extends AbstractContextComponent {
52

53
    private static final int LONG_LITERAL_LENGTH = 100;
54
    private static final Logger logger = LoggerFactory.getLogger(ReadonlyItem.class);
×
55
    private static final ValueFactory vf = SimpleValueFactory.getInstance();
×
56

57
    private IModel<String> model;
58
    private String prefix;
59
    private ExternalLink linkComp;
60
    private Label extraComp, languageComp, datatypeComp;
61
    private IModel<String> extraModel, languageModel, datatypeModel;
62
    private IRI iri;
63
    private RestrictedChoice restrictedChoice;
64
    private Label showMoreLabelLiteral, showMoreLabelHTML;
65
    private final Template template;
66

67
    /**
68
     * Constructor for ReadonlyItem.
69
     *
70
     * @param id              the component id
71
     * @param parentId        the parent id (e.g., "subj", "obj")
72
     * @param iriP            the IRI of the item
73
     * @param statementPartId the statement part ID
74
     * @param rg              the repetition group
75
     */
76
    public ReadonlyItem(String id, String parentId, final IRI iriP, IRI statementPartId, final RepetitionGroup rg) {
77
        super(id, rg.getContext());
×
78
        this.iri = iriP;
×
79
        template = context.getTemplate();
×
80
        model = (IModel<String>) context.getComponentModels().get(iri);
×
81
        boolean modelIsNew = false;
×
82
        if (model == null) {
×
83
            model = Model.of("");
×
84
            context.getComponentModels().put(iri, model);
×
85
            modelIsNew = true;
×
86
        }
87
        String postfix = Utils.getUriPostfix(iri);
×
88
        if (modelIsNew && context.hasParam(postfix)) {
×
89
            model.setObject(context.getParam(postfix));
×
90
        }
91
        if (model.getObject().isEmpty() && template.isRootNanopubPlaceholder(iri) && context.getReferenceNanopub() == null) {
×
92
            model.setObject(LocalUri.of("nanopub").stringValue());
×
93
        }
94

95
        final Map<String, String> foafNameMap;
96
        if (context.getExistingNanopub() == null) {
×
97
            foafNameMap = new HashMap<>();
×
98
        } else {
99
            foafNameMap = Utils.getFoafNameMap(context.getExistingNanopub());
×
100
        }
101

102
        prefix = template.getPrefix(iri);
×
103
        if (prefix == null) prefix = "";
×
104
        if (template.isRestrictedChoicePlaceholder(iri)) {
×
105
            restrictedChoice = new RestrictedChoice(iri, context);
×
106
        }
107
        add(new Label("prefix", new Model<String>() {
×
108

109
            @Override
110
            public String getObject() {
111
                String prefixLabel = template.getPrefixLabel(iri);
×
112
                String v = getFullValue();
×
113
                if (prefixLabel == null || IndividualAgent.isUser(v) || foafNameMap.containsKey(v)) {
×
114
                    return "";
×
115
                } else if (prefixLabel.matches("https?://.*")) {
×
116
                    // Hide prefix label that leaks through as a raw URL.
117
                    return "";
×
118
                } else {
119
                    if (!prefixLabel.isEmpty() && parentId.equals("subj")) {
×
120
                        // Capitalize first letter of label if at subject position:
121
                        prefixLabel = prefixLabel.substring(0, 1).toUpperCase() + prefixLabel.substring(1);
×
122
                    }
123
                    return prefixLabel;
×
124
                }
125
            }
126

127
        }));
128

129
        linkComp = new ExternalLink("link", new Model<String>() {
×
130

131
            @Override
132
            public String getObject() {
133
                String obj = getFullValue();
×
134
                if (obj == null) return "";
×
135
                if (obj.equals(LocalUri.of("nanopub").stringValue())) {
×
136
                    if (context.getExistingNanopub() != null) {
×
137
                        obj = context.getExistingNanopub().getUri().stringValue();
×
138
                        return ExplorePage.MOUNT_PATH + "?id=" + URLEncoder.encode(obj, Charsets.UTF_8);
×
139
                    } else {
140
                        return "";
×
141
                    }
142
                } else if (obj.equals(LocalUri.of("assertion").stringValue())) {
×
143
                    if (context.getExistingNanopub() != null) {
×
144
                        obj = context.getExistingNanopub().getAssertionUri().stringValue();
×
145
                        return ExplorePage.MOUNT_PATH + "?id=" + URLEncoder.encode(obj, Charsets.UTF_8);
×
146
                    } else {
147
                        return "";
×
148
                    }
149
                } else if (obj.matches("https?://.+")) {
×
150
                    return NanodashLink.getPageUrl(obj);
×
151
                } else {
152
                    return "";
×
153
                }
154
            }
155

156
        }, new Model<String>() {
×
157

158
            @Override
159
            public String getObject() {
160
                String obj = getFullValue();
×
161
                if (obj != null && obj.equals(LocalUri.of("nanopub").stringValue())) {
×
162
                    return "this nanopublication";
×
163
                }
164
                if (obj != null && obj.equals(LocalUri.of("assertion").stringValue())) {
×
165
                    return context.getType() == ContextType.ASSERTION ? "this assertion" : "the assertion above";
×
166
                }
167
                if (obj != null && obj.matches("https?://.+")) {
×
168
                    IRI objIri = vf.createIRI(obj);
×
169
                    if (iri.equals(NTEMPLATE.CREATOR_PLACEHOLDER)) {
×
170
                        // TODO We might want to introduce a "(you)" flag here at some point
171
                        return User.getShortDisplayName(objIri);
×
172
                    } else if (isAssertionValue(objIri)) {
×
173
                        if (context.getType() == ContextType.ASSERTION) {
×
174
                            return "this assertion";
×
175
                        } else {
176
                            return "the assertion above";
×
177
                        }
178
                    } else if (isNanopubValue(objIri)) {
×
179
                        return "this nanopublication";
×
180
                    } else if (IndividualAgent.isUser(obj)) {
×
181
                        return User.getShortDisplayName(objIri);
×
182
                    } else if (foafNameMap.containsKey(obj)) {
×
183
                        return foafNameMap.get(obj);
×
184
                    }
185
                    Space space = SpaceRepository.get().findById(obj);
×
186
                    if (space != null) return space.getLabel();
×
187
                    space = SpaceRepository.get().findByAltId(obj);
×
188
                    if (space != null) return space.getLabel();
×
189
                    MaintainedResource mr = MaintainedResourceRepository.get().findById(obj);
×
190
                    if (mr != null) return mr.getLabel();
×
191
                    return getLabelString(objIri);
×
192
                }
193
                return obj;
×
194
            }
195

196
        });
197
        if (template.isIntroducedResource(iri) || template.isEmbeddedResource(iri)) {
×
198
            linkComp.add(AttributeAppender.append("class", "introduced"));
×
199
        }
200
        add(linkComp);
×
201
        add(new Label("description", new Model<String>() {
×
202

203
            @Override
204
            public String getObject() {
205
                String obj = getFullValue();
×
206
                if (obj != null && obj.matches("https?://.+")) {
×
207
                    IRI objIri = vf.createIRI(obj);
×
208
                    if (isAssertionValue(objIri)) {
×
209
                        return "This is the identifier for the assertion of this nanopublication.";
×
210
                    } else if (isNanopubValue(objIri)) {
×
211
                        return "This is the identifier for this whole nanopublication.";
×
212
                    } else if (context.isReadOnly() && obj.startsWith(context.getExistingNanopub().getUri().stringValue())) {
×
213
                        return "This is a local identifier minted within the nanopublication.";
×
214
                    }
215
                    String labelString = getLabelString(objIri);
×
216
                    String description = "";
×
217
                    if (labelString.contains(" - ")) description = labelString.replaceFirst("^.* - ", "");
×
218
                    return description;
×
219
                } else if (obj != null && obj.startsWith("\"")) {
×
220
                    return "(this is a literal)";
×
221
                }
222
                return "";
×
223
            }
224

225
        }));
226
        Model<String> uriModel = new Model<String>() {
×
227

228
            @Override
229
            public String getObject() {
230
                String obj = getFullValue();
×
231
                if (obj != null && obj.startsWith("\"")) return "";
×
232
                if (isAssertionValue(obj)) {
×
233
                    return getAssertionValue();
×
234
                } else if (isNanopubValue(obj)) {
×
235
                    return getNanopubValue();
×
236
                }
237
                return obj;
×
238
            }
239

240
        };
241
        add(Utils.getUriLink("uri", uriModel));
×
242
        extraModel = Model.of("");
×
243
        extraComp = new Label("extra", extraModel);
×
244
        extraComp.setVisible(false);
×
245
        add(extraComp);
×
246
        languageModel = Model.of("");
×
247
        languageComp = new Label("language", languageModel);
×
248
        languageComp.setVisible(false);
×
249
        add(languageComp);
×
250
        datatypeModel = Model.of("");
×
251
        datatypeComp = new Label("datatype", datatypeModel);
×
252
        datatypeComp.setVisible(false);
×
253
        add(datatypeComp);
×
254

255
        showMoreLabelLiteral = new Label("show-more-literal", "");
×
256
        add(showMoreLabelLiteral);
×
257
        showMoreLabelLiteral.setVisible(false);
×
258

259
        showMoreLabelHTML = new Label("show-more-html", "");
×
260
        add(showMoreLabelHTML);
×
261
        showMoreLabelHTML.setVisible(false);
×
262
    }
×
263

264
    /**
265
     * {@inheritDoc}
266
     */
267
    @Override
268
    public void fillFinished() {
269
        String obj = getFullValue();
×
270
        if (obj != null) {
×
271
            if (isAssertionValue(obj)) {
×
272
                linkComp.add(new AttributeAppender("class", "this-assertion"));
×
273
            } else if (isNanopubValue(obj)) {
×
274
                linkComp.add(new AttributeAppender("class", "this-nanopub"));
×
275
            } else if (context.getExistingNanopub() != null) {
×
276
                Nanopub np = context.getExistingNanopub();
×
277
                if (NanopubUtils.getIntroducedIriIds(np).contains(obj) || NanopubUtils.getEmbeddedIriIds(np).contains(obj)) {
×
278
                    linkComp.add(AttributeAppender.append("class", "introduced"));
×
279
                }
280
            }
281
        }
282
    }
×
283

284
    /**
285
     * {@inheritDoc}
286
     */
287
    @Override
288
    public void finalizeValues() {
289
    }
×
290

291
    private String getLabelString(IRI iri) {
292
        if (template.getLabel(iri) != null) {
×
293
            return template.getLabel(iri).replaceFirst(" - .*$", "");
×
294
        } else if (context.getLabel(iri) != null) {
×
295
            return context.getLabel(iri).replaceFirst(" - .*$", "");
×
296
        } else {
297
            return Utils.getShortNameFromURI(iri.stringValue());
×
298
        }
299
    }
300

301
    /**
302
     * {@inheritDoc}
303
     */
304
    @Override
305
    public void removeFromContext() {
306
        // Nothing to be done here.
307
    }
×
308

309
    private String getFullValue() {
310
        String s = model.getObject();
×
311
        if (s == null) return null;
×
312
        if (template.isAutoEscapePlaceholder(iri)) {
×
313
            s = Utils.urlEncode(s);
×
314
        }
315
        if (!prefix.isEmpty()) {
×
316
            s = prefix + s;
×
317
        }
318
        return s;
×
319
    }
320

321
    private boolean isNanopubValue(Object obj) {
322
        if (obj == null) return false;
×
323
        if (obj.toString().equals(LocalUri.of("nanopub").stringValue())) return true;
×
324
        // A fillSource match means a *prior* nanopub (supersede/derive), not "this" one — don't label it as such.
325
        if (context.getExistingNanopub() == null) return false;
×
326
        return obj.toString().equals(context.getExistingNanopub().getUri().stringValue());
×
327
    }
328

329
    private String getNanopubValue() {
330
        Nanopub ref = context.getReferenceNanopub();
×
331
        if (ref != null) {
×
332
            return ref.getUri().stringValue();
×
333
        } else {
334
            return LocalUri.of("nanopub").stringValue();
×
335
        }
336
    }
337

338
    private boolean isAssertionValue(Object obj) {
339
        if (obj == null) return false;
×
340
        if (obj.toString().equals(LocalUri.of("assertion").stringValue())) return true;
×
341
        // A fillSource match means a *prior* assertion (supersede/derive), not "this" one.
342
        if (context.getExistingNanopub() == null) return false;
×
343
        return obj.toString().equals(context.getExistingNanopub().getAssertionUri().stringValue());
×
344
    }
345

346
    private String getAssertionValue() {
347
        Nanopub ref = context.getReferenceNanopub();
×
348
        if (ref != null) {
×
349
            return ref.getAssertionUri().stringValue();
×
350
        } else {
351
            return LocalUri.of("assertion").stringValue();
×
352
        }
353
    }
354

355
    /**
356
     * {@inheritDoc}
357
     */
358
    @Override
359
    public boolean isUnifiableWith(Value v) {
360
        if (v == null) return true;
×
361
        if (v instanceof IRI) {
×
362
            String vs = v.stringValue();
×
363
            if (vs.equals(LocalUri.of("nanopub").stringValue())) {
×
364
                vs = getNanopubValue();
×
365
            } else if (vs.equals(LocalUri.of("assertion").stringValue())) {
×
366
                vs = getAssertionValue();
×
367
            }
368
            if (vs.startsWith(prefix)) vs = vs.substring(prefix.length());
×
369
//                        if (Utils.isLocalURI(vs)) vs = vs.replaceFirst("^" + LocalUri.PREFIX, "");
370
            if (template.isAutoEscapePlaceholder(iri)) {
×
371
                vs = Utils.urlDecode(vs);
×
372
            }
373
            Validatable<String> validatable = new Validatable<>(vs);
×
374
//                        if (template.isLocalResource(iri) && !Utils.isUriPostfix(vs)) {
375
//                                vs = Utils.getUriPostfix(vs);
376
//                        }
377
            new Validator().validate(validatable);
×
378
            if (!validatable.isValid()) {
×
379
                return false;
×
380
            }
381
            if (model.getObject().isEmpty()) {
×
382
                return true;
×
383
            }
384
            return vs.equals(model.getObject());
×
385
        } else if (v instanceof Literal vL) {
×
386
            if (template.getRegex(iri) != null && !v.stringValue().matches(template.getRegex(iri))) {
×
387
                return false;
×
388
            }
389
            String languagetag = template.getLanguageTag(iri);
×
390
            IRI datatype = template.getDatatype(iri);
×
391
            if (languagetag != null) {
×
392
                if (vL.getLanguage().isEmpty() || !Literals.normalizeLanguageTag(vL.getLanguage().get()).equals(languagetag)) {
×
393
                    return false;
×
394
                }
395
            } else if (datatype != null) {
×
396
                if (!vL.getDatatype().equals(datatype)) {
×
397
                    return false;
×
398
                }
399
            }
400
            if (linkComp.getDefaultModelObject() == null || linkComp.getDefaultModelObject().toString().isEmpty()) {
×
401
                return true;
×
402
            }
403
            return linkComp.getDefaultModelObject().equals("\"" + v.stringValue() + "\"");
×
404
        }
405
        return false;
×
406
    }
407

408
    /**
409
     * {@inheritDoc}
410
     */
411
    @Override
412
    public void unifyWith(Value v) throws UnificationException {
413
        if (v == null) return;
×
414
        String vs = v.stringValue();
×
415
        if (!isUnifiableWith(v)) {
×
416
            logger.error("Cannot unify {}", v);
×
417
            throw new UnificationException(vs);
×
418
        }
419
        if (v instanceof IRI) {
×
420
            if (vs.equals(LocalUri.of("nanopub").stringValue())) {
×
421
                vs = getNanopubValue();
×
422
            } else if (vs.equals(LocalUri.of("assertion").stringValue())) {
×
423
                vs = getAssertionValue();
×
424
            }
425
            if (!prefix.isEmpty() && vs.startsWith(prefix)) {
×
426
                vs = vs.substring(prefix.length());
×
427
                // With read-only items, we don't need preliminary local identifiers:
428
//                        } else if (Utils.isLocalURI(vs)) {
429
//                                vs = vs.replaceFirst("^" + LocalUri.PREFIX, "");
430
//                        } else if (template.isLocalResource(iri) && !Utils.isUriPostfix(vs)) {
431
//                                vs = Utils.getUriPostfix(vs);
432
            }
433
            if (template.isAutoEscapePlaceholder(iri)) {
×
434
                vs = Utils.urlDecode(vs);
×
435
            }
436
            model.setObject(vs);
×
437
        } else if (v instanceof Literal vL) {
×
438
            if (vs.length() >= LONG_LITERAL_LENGTH) {
×
439
                linkComp.add(AttributeAppender.append("class", "long-literal collapsed"));
×
440
                showMoreLabelLiteral.setVisible(true);
×
441
            }
442
            if (vL.getLanguage().isPresent()) {
×
443
                model.setObject("\"" + vs + "\"");
×
444
                languageModel.setObject("(" + Literals.normalizeLanguageTag(vL.getLanguage().get()) + ")");
×
445
                languageComp.setVisible(true);
×
446
            } else if (!vL.getDatatype().equals(XSD.STRING)) {
×
447
                model.setObject("\"" + vs + "\"");
×
448
                datatypeModel.setObject("(" + vL.getDatatype().stringValue().replace(XSD.NAMESPACE, "xsd:") + ")");
×
449
                datatypeComp.setVisible(true);
×
450
            } else {
451
                model.setObject("\"" + vs + "\"");
×
452
            }
453
            if (Utils.looksLikeHtml(vs)) {
×
454
                linkComp.setVisible(false);
×
455
                extraModel.setObject(Utils.sanitizeHtml(vs));
×
456
                extraComp.setEscapeModelStrings(false);
×
457
                extraComp.setVisible(true);
×
458
                showMoreLabelLiteral.setVisible(false);
×
459
                showMoreLabelHTML.setVisible(true);
×
460
            }
461
        }
462
    }
×
463

464
    /**
465
     * Validator class for validating the input.
466
     */
467
    protected class Validator extends InvalidityHighlighting implements IValidator<String> {
468

469
        /**
470
         * Default constructor for Validator.
471
         */
472
        public Validator() {
×
473
        }
×
474

475
        @Override
476
        public void validate(IValidatable<String> s) {
477
            String sv = s.getValue();
×
478
            String p = prefix;
×
479
            if (template.isAutoEscapePlaceholder(iri)) {
×
480
                sv = Utils.urlEncode(sv);
×
481
            }
482
            if (sv.matches("https?://.+")) {
×
483
                p = "";
×
484
            } else if (sv.contains(":")) {
×
485
                s.error(new ValidationError("Colon character is not allowed in postfix"));
×
486
            }
487
            String iriString = p + sv;
×
488
            if (iriString.matches("[^:# ]+")) {
×
489
                p = LocalUri.PREFIX;
×
490
                iriString = p + sv;
×
491
            }
492
            try {
493
                ParsedIRI piri = new ParsedIRI(iriString);
×
494
                if (!piri.isAbsolute()) {
×
495
                    s.error(new ValidationError("IRI not well-formed"));
×
496
                }
497
                if (p.isEmpty() && !Utils.isLocalURI(sv) && !sv.matches("https?://.+")) {
×
498
                    s.error(new ValidationError("Only http(s):// IRIs are allowed here"));
×
499
                }
500
            } catch (URISyntaxException ex) {
×
501
                s.error(new ValidationError("IRI not well-formed"));
×
502
            }
×
503
            String regex = template.getRegex(iri);
×
504
            if (regex != null) {
×
505
                if (!sv.matches(regex)) {
×
506
                    s.error(new ValidationError("Value '" + sv + "' doesn't match the pattern '" + regex + "'"));
×
507
                }
508
            }
509
            if (template.isRestrictedChoicePlaceholder(iri)) {
×
510
                if (!restrictedChoice.getPossibleValues().contains(iriString) && !restrictedChoice.hasPossibleRefValues()) {
×
511
                    // not checking the possible ref values can overgenerate, but normally works
512
                    s.error(new ValidationError("Invalid choice"));
×
513
                }
514
            }
515
            if (template.isExternalUriPlaceholder(iri)) {
×
516
                if (!iriString.matches("https?://.+")) {
×
517
                    s.error(new ValidationError("Not an external IRI"));
×
518
                }
519
            }
520
            if (template.isTrustyUriPlaceholder(iri)) {
×
521
                if (!TrustyUriUtils.isPotentialTrustyUri(iriString)) {
×
522
                    s.error(new ValidationError("Not a trusty URI"));
×
523
                }
524
            }
525
            if (iri.equals(NTEMPLATE.CREATOR_PLACEHOLDER) && context.getExistingNanopub() != null) {
×
526
                boolean found = false;
×
527
                for (IRI creator : SimpleCreatorPattern.getCreators(context.getExistingNanopub())) {
×
528
                    if (creator.stringValue().equals(iriString)) {
×
529
                        found = true;
×
530
                        break;
×
531
                    }
532
                }
×
533
                if (!found) {
×
534
                    s.error(new ValidationError("Not a creator of nanopub"));
×
535
                }
536
            }
537
        }
×
538

539
    }
540

541
    /**
542
     * <p>toString.</p>
543
     *
544
     * @return a {@link java.lang.String} object
545
     */
546
    public String toString() {
547
        return "[read-only IRI item: " + iri + "]";
×
548
    }
549

550
}
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