35
35
import javax .annotation .Nullable ;
36
36
import javax .annotation .concurrent .Immutable ;
37
37
38
+ import org .apache .pekko .actor .ActorSystem ;
38
39
import org .eclipse .ditto .base .model .exceptions .DittoRuntimeException ;
39
40
import org .eclipse .ditto .base .model .headers .DittoHeaderDefinition ;
40
41
import org .eclipse .ditto .base .model .headers .DittoHeaders ;
98
99
import org .eclipse .ditto .wot .model .WotThingModelInvalidException ;
99
100
import org .eclipse .ditto .wot .model .WotThingModelPlaceholderUnresolvedException ;
100
101
101
- import org .apache .pekko .actor .ActorSystem ;
102
-
103
102
/**
104
103
* Default Ditto specific implementation of {@link WotThingDescriptionGenerator}.
105
104
*/
@@ -311,9 +310,9 @@ private void removeTmRequired(final ThingModel.Builder builder) {
311
310
private void addBase (final ThingDescription .Builder thingDescriptionBuilder ,
312
311
final ThingId thingId , @ Nullable final String featureId ) {
313
312
if (null != featureId ) {
314
- final String featurePath = "/ features/" + featureId ;
315
- thingDescriptionBuilder .setId (IRI .of ("urn:" + thingId + featurePath ))
316
- .setBase (IRI .of (buildThingIdBasePath (thingId ) + featurePath ));
313
+ final String featurePath = "features/" + featureId ;
314
+ thingDescriptionBuilder .setId (IRI .of ("urn:" + thingId + "/" + featurePath ))
315
+ .setBase (IRI .of (buildThingIdBasePath (thingId ) + featurePath + "/" ));
317
316
} else {
318
317
thingDescriptionBuilder .setId (IRI .of ("urn:" + thingId ))
319
318
.setBase (IRI .of (buildThingIdBasePath (thingId )));
@@ -331,7 +330,7 @@ private void addInstanceVersion(final ThingDescription.Builder thingDescriptionB
331
330
}
332
331
333
332
private String buildThingIdBasePath (final ThingId thingId ) {
334
- return toThingDescriptionConfig .getBasePrefix () + "/api/2/things/" + thingId ;
333
+ return toThingDescriptionConfig .getBasePrefix () + "/api/2/things/" + thingId + "/" ;
335
334
}
336
335
337
336
private void addThingDescriptionLinks (final ThingDescription .Builder tdBuilder , final URL tmUrl ,
@@ -372,7 +371,7 @@ private void convertThingDescriptionTmSubmodelLinksToItems(final ThingDescriptio
372
371
return Link .newBuilder ()
373
372
.setRel ("item" )
374
373
.setType (ContentType .APPLICATION_TD_JSON .getValue ())
375
- .setHref (IRI .of ("/ features/" + link .getValue (TM_SUBMODEL_INSTANCE_NAME )
374
+ .setHref (IRI .of ("features/" + link .getValue (TM_SUBMODEL_INSTANCE_NAME )
376
375
.filter (JsonValue ::isString )
377
376
.map (JsonValue ::asString )
378
377
.orElseThrow (() -> WotThingModelInvalidException
@@ -436,11 +435,12 @@ private void generateRootForms(final ThingModel thingModel,
436
435
DITTO_FIELDS_URI_VARIABLE ,
437
436
DittoHeaderDefinition .CHANNEL .getKey (),
438
437
DittoHeaderDefinition .TIMEOUT .getKey ());
438
+ final String hrefWithoutLeadingSlash = propertiesPath .toString ().substring (1 );
439
439
if (thingModelForms .isPresent ()) {
440
440
tdBuilder .setForms (thingModelForms .get ()
441
441
.stream ()
442
442
.map (rfe -> RootFormElement .fromJson (rfe .toBuilder ()
443
- .setHref (IRI .of (propertiesPath ))
443
+ .setHref (IRI .of (hrefWithoutLeadingSlash ))
444
444
.setAdditionalResponses (provideAdditionalResponses ())
445
445
.build ()
446
446
))
@@ -449,27 +449,27 @@ private void generateRootForms(final ThingModel thingModel,
449
449
} else {
450
450
tdBuilder .setForms (List .of (
451
451
buildRootFormElement (SingleRootFormElementOp .READALLPROPERTIES ,
452
- propertiesPath + readUriVariablesParams ,
452
+ hrefWithoutLeadingSlash + readUriVariablesParams ,
453
453
"GET"
454
454
),
455
455
buildRootFormElement (SingleRootFormElementOp .READMULTIPLEPROPERTIES ,
456
- propertiesPath + readMultiplePropertiesUriVariablesParams ,
456
+ hrefWithoutLeadingSlash + readMultiplePropertiesUriVariablesParams ,
457
457
"GET"
458
458
),
459
459
buildRootFormElement (SingleRootFormElementOp .WRITEALLPROPERTIES ,
460
- propertiesPath + writeUriVariablesParams ,
460
+ hrefWithoutLeadingSlash + writeUriVariablesParams ,
461
461
"PUT"
462
462
),
463
463
buildRootFormElement (SingleRootFormElementOp .WRITEMULTIPLEPROPERTIES ,
464
- propertiesPath + writeUriVariablesParams ,
464
+ hrefWithoutLeadingSlash + writeUriVariablesParams ,
465
465
"PATCH" ,
466
466
builder -> builder .setContentType (ContentType .APPLICATION_MERGE_PATCH_JSON .getValue ())
467
467
),
468
468
buildRootFormElement (List .of (
469
469
SingleRootFormElementOp .OBSERVEALLPROPERTIES ,
470
470
SingleRootFormElementOp .UNOBSERVEALLPROPERTIES
471
471
),
472
- propertiesPath ,
472
+ hrefWithoutLeadingSlash ,
473
473
"GET" ,
474
474
builder -> builder
475
475
.setSubprotocol (SUBPROTOCOL_SSE )
@@ -479,7 +479,7 @@ private void generateRootForms(final ThingModel thingModel,
479
479
SingleRootFormElementOp .SUBSCRIBEALLEVENTS ,
480
480
SingleRootFormElementOp .UNSUBSCRIBEALLEVENTS
481
481
),
482
- JsonPointer . of ( "/ outbox/messages") ,
482
+ " outbox/messages" ,
483
483
"GET" ,
484
484
builder -> builder
485
485
.setSubprotocol (SUBPROTOCOL_SSE )
@@ -512,7 +512,7 @@ private RootFormElement buildRootFormElement(final SingleRootFormElementOp op,
512
512
}
513
513
514
514
private RootFormElement buildRootFormElement (final Collection <SingleRootFormElementOp > ops ,
515
- final JsonPointer hrefPointer ,
515
+ final CharSequence hrefPointer ,
516
516
final String htvMethodName ,
517
517
final Consumer <RootFormElement .Builder > builderConsumer
518
518
) {
@@ -557,11 +557,12 @@ private void generatePropertiesForms(final ThingModel thingModel,
557
557
final String readUriVariablesParams = provideUriVariablesBag (
558
558
DittoHeaderDefinition .CHANNEL .getKey (),
559
559
DittoHeaderDefinition .TIMEOUT .getKey ());
560
+ final String hrefWithoutLeadingSlash = propertyHref .toString ().substring (1 );
560
561
return property .getForms ()
561
562
.map (propertyFormElements -> property .toBuilder ()
562
563
.setForms (PropertyForms .of (propertyFormElements .stream ()
563
564
.map (pfe -> pfe .toBuilder ()
564
- .setHref (IRI .of (propertyHref ))
565
+ .setHref (IRI .of (hrefWithoutLeadingSlash ))
565
566
.setAdditionalResponses (provideAdditionalResponses ())
566
567
.build ()
567
568
)
@@ -573,18 +574,18 @@ private void generatePropertiesForms(final ThingModel thingModel,
573
574
final List <PropertyFormElement > formElements = new ArrayList <>();
574
575
if (!property .isWriteOnly ()) {
575
576
formElements .add (buildPropertyFormElement (SinglePropertyFormElementOp .READPROPERTY ,
576
- propertyHref + readUriVariablesParams ,
577
+ hrefWithoutLeadingSlash + readUriVariablesParams ,
577
578
"GET"
578
579
));
579
580
}
580
581
581
582
if (!property .isReadOnly ()) {
582
583
formElements .add (buildPropertyFormElement (SinglePropertyFormElementOp .WRITEPROPERTY ,
583
- propertyHref + writeUriVariablesParams ,
584
+ hrefWithoutLeadingSlash + writeUriVariablesParams ,
584
585
"PUT"
585
586
));
586
587
formElements .add (buildPropertyFormElement (SinglePropertyFormElementOp .WRITEPROPERTY ,
587
- propertyHref + writeUriVariablesParams ,
588
+ hrefWithoutLeadingSlash + writeUriVariablesParams ,
588
589
"PATCH" ,
589
590
builder -> builder
590
591
.setContentType (
@@ -599,7 +600,7 @@ private void generatePropertiesForms(final ThingModel thingModel,
599
600
SinglePropertyFormElementOp .OBSERVEPROPERTY ,
600
601
SinglePropertyFormElementOp .UNOBSERVEPROPERTY
601
602
),
602
- propertyHref ,
603
+ hrefWithoutLeadingSlash ,
603
604
"GET" ,
604
605
builder -> builder
605
606
.setSubprotocol (SUBPROTOCOL_SSE )
@@ -725,7 +726,7 @@ private void generateActionsForms(final ThingModel thingModel,
725
726
.map (actions -> actions .map (actionEntry -> {
726
727
final String actionName = actionEntry .getKey ();
727
728
final Action action = actionEntry .getValue ();
728
- final JsonPointer actionHref = JsonPointer . of ( "/ inbox/messages/" + actionName ) ;
729
+ final String actionHrefWithoutLeadingSlash = " inbox/messages/" + actionName ;
729
730
final String uriVariablesParams = provideUriVariablesBag (
730
731
DittoHeaderDefinition .TIMEOUT .getKey (),
731
732
DittoHeaderDefinition .RESPONSE_REQUIRED .getKey ());
@@ -734,7 +735,7 @@ private void generateActionsForms(final ThingModel thingModel,
734
735
.setSynchronous (true )
735
736
.setForms (ActionForms .of (actionFormElements .stream ()
736
737
.map (afe -> afe .toBuilder ()
737
- .setHref (IRI .of (actionHref ))
738
+ .setHref (IRI .of (actionHrefWithoutLeadingSlash ))
738
739
.setAdditionalResponses (provideAdditionalResponses ())
739
740
.build ()
740
741
)
@@ -746,7 +747,7 @@ private void generateActionsForms(final ThingModel thingModel,
746
747
.setSynchronous (true )
747
748
.setForms (ActionForms .of (List .of (
748
749
buildActionFormElement (SingleActionFormElementOp .INVOKEACTION ,
749
- actionHref + uriVariablesParams )
750
+ actionHrefWithoutLeadingSlash + uriVariablesParams )
750
751
)))
751
752
.build ()
752
753
);
@@ -788,12 +789,12 @@ private void generateEventsForms(final ThingModel thingModel, final ThingDescrip
788
789
.map (events -> events .map (eventEntry -> {
789
790
final String eventName = eventEntry .getKey ();
790
791
final Event event = eventEntry .getValue ();
791
- final JsonPointer eventHref = JsonPointer . of ( "/ outbox/messages/" + eventName ) ;
792
+ final String eventHrefWithoutLeadingSlash = " outbox/messages/" + eventName ;
792
793
return event .getForms ()
793
794
.map (eventFormElements -> event .toBuilder ()
794
795
.setForms (EventForms .of (eventFormElements .stream ()
795
796
.map (efe -> efe .toBuilder ()
796
- .setHref (IRI .of (eventHref ))
797
+ .setHref (IRI .of (eventHrefWithoutLeadingSlash ))
797
798
.setAdditionalResponses (provideAdditionalResponses ())
798
799
.build ()
799
800
)
@@ -803,7 +804,8 @@ private void generateEventsForms(final ThingModel thingModel, final ThingDescrip
803
804
)
804
805
.orElseGet (() -> event .toBuilder ()
805
806
.setForms (EventForms .of (List .of (
806
- buildEventFormElement (SingleEventFormElementOp .SUBSCRIBEEVENT , eventHref )
807
+ buildEventFormElement (SingleEventFormElementOp .SUBSCRIBEEVENT ,
808
+ eventHrefWithoutLeadingSlash )
807
809
)))
808
810
.build ()
809
811
);
@@ -814,7 +816,7 @@ private void generateEventsForms(final ThingModel thingModel, final ThingDescrip
814
816
}
815
817
816
818
private EventFormElement buildEventFormElement (final SingleEventFormElementOp op ,
817
- final JsonPointer hrefPointer
819
+ final CharSequence hrefPointer
818
820
) {
819
821
return EventFormElement .newBuilder ()
820
822
.setOp (op )
@@ -887,7 +889,8 @@ private Optional<JsonValue> resolvePlaceholder(final String value, @Nullable fin
887
889
final String placeholderToResolve = matcher .group (TM_PLACEHOLDER_PL_GROUP ).trim ();
888
890
if (null != modelPlaceholders ) {
889
891
return modelPlaceholders .getValue (placeholderToResolve )
890
- .or (() -> Optional .ofNullable (toThingDescriptionConfig .getPlaceholders ().get (placeholderToResolve )));
892
+ .or (() -> Optional .ofNullable (
893
+ toThingDescriptionConfig .getPlaceholders ().get (placeholderToResolve )));
891
894
} else {
892
895
return Optional .ofNullable (toThingDescriptionConfig .getPlaceholders ().get (placeholderToResolve ));
893
896
}
0 commit comments