From d2c4cd4dd8d7d3f49068e4cfe3c82201a3450ee5 Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Wed, 20 Nov 2024 17:22:02 +0100 Subject: [PATCH 1/3] Add LDObject.tryFromDynamicObj + tests --- src/ROCrate/LDObject.fs | 17 ++++++++- tests/ROCrate/LDObject.Tests.fs | 64 +++++++++++++++++++++++++++------ 2 files changed, 69 insertions(+), 12 deletions(-) diff --git a/src/ROCrate/LDObject.fs b/src/ROCrate/LDObject.fs index bf001ba7..5dcc8745 100644 --- a/src/ROCrate/LDObject.fs +++ b/src/ROCrate/LDObject.fs @@ -56,4 +56,19 @@ type LDObject(id:string, schemaType: string, ?additionalType) = member this.RemoveContext() = this.RemoveProperty("@context") - static member removeContext () = fun (roc: #LDObject) -> roc.RemoveContext() \ No newline at end of file + static member removeContext () = fun (roc: #LDObject) -> roc.RemoveContext() + + static member tryFromDynamicObj (dynObj: DynamicObj) = + match + DynObj.tryGetTypedPropertyValue("@type") dynObj, + DynObj.tryGetTypedPropertyValue("@id") dynObj, + DynObj.tryGetTypedPropertyValue("additionalType") dynObj + with + | (Some schemaType), (Some id), at -> + let roc = new LDObject(id, schemaType, ?additionalType = at) + match DynObj.tryGetTypedPropertyValue("@context") dynObj with + | Some context -> roc.SetContext(context) + | _ -> () + Some roc + | _ -> None + \ No newline at end of file diff --git a/tests/ROCrate/LDObject.Tests.fs b/tests/ROCrate/LDObject.Tests.fs index 658c7e1d..98b3b9ea 100644 --- a/tests/ROCrate/LDObject.Tests.fs +++ b/tests/ROCrate/LDObject.Tests.fs @@ -6,8 +6,19 @@ open DynamicObj open TestingUtils open Common +let context = + new LDContext() + |> DynObj.withProperty "more" "context" + let mandatory_properties = LDObject("LDObject_mandatory_properties_id", "someType") +let mandatory_properties_with_context = + LDObject("LDObject_mandatory_properties_id", "someType") + |> DynObj.withProperty "@context" context + let all_properties = LDObject("LDObject_all_properties_id", "someType", additionalType = "additionalType") +let all_properties_with_context = + LDObject("LDObject_all_properties_id", "someType", additionalType = "additionalType") + |> DynObj.withProperty "@context" (context.CopyDynamicProperties()) let tests_profile_object_is_valid = testList "constructed properties" [ testList "mandatory properties" [ @@ -58,19 +69,50 @@ let tests_instance_methods = testSequenced ( let tests_static_methods = testSequenced ( testList "static methods" [ + testList "context" [ + let context = new LDContext() + context.SetProperty("more", "context") - let context = new LDContext() - context.SetProperty("more", "context") + testCase "can set context" <| fun _ -> + LDObject.setContext context mandatory_properties + Expect.LDObjectHasDynamicProperty "@context" context mandatory_properties + testCase "can get context" <| fun _ -> + let ctx = LDObject.tryGetContext() mandatory_properties + Expect.equal ctx (Some context) "context was not set correctly" + testCase "can remove context" <| fun _ -> + LDObject.removeContext() mandatory_properties |> ignore + Expect.isNone (DynObj.tryGetTypedPropertyValue "@context" mandatory_properties) "context was not removed correctly" + ] + testList "tryFromDynamicObj" [ + let compatibleDynObj = + let tmp = DynamicObj() + tmp + |> DynObj.withProperty "@type" "someType" + |> DynObj.withProperty "@id" "LDObject_all_properties_id" + |> DynObj.withProperty "additionalType" "additionalType" - testCase "can set context" <| fun _ -> - LDObject.setContext context mandatory_properties - Expect.LDObjectHasDynamicProperty "@context" context mandatory_properties - testCase "can get context" <| fun _ -> - let ctx = LDObject.tryGetContext() mandatory_properties - Expect.equal ctx (Some context) "context was not set correctly" - testCase "can remove context" <| fun _ -> - LDObject.removeContext() mandatory_properties |> ignore - Expect.isNone (DynObj.tryGetTypedPropertyValue "@context" mandatory_properties) "context was not removed correctly" + let compatibleDynObjWithContext = + let tmp = DynamicObj() + tmp + |> DynObj.withProperty "@type" "someType" + |> DynObj.withProperty "@id" "LDObject_all_properties_id" + |> DynObj.withProperty "additionalType" "additionalType" + |> DynObj.withProperty "@context" context + + let incompatibleDynObj = + let tmp = DynamicObj() + tmp + |> DynObj.withProperty "@type" "someType" + + testCase "can convert compatible DynObj to LDObject" <| fun _ -> + let roc = Expect.wantSome (LDObject.tryFromDynamicObj compatibleDynObj) "LDObject.tryFromDynamicObj did not return Some" + Expect.equal roc all_properties "LDObject was not created correctly from compatible DynamicObj" + testCase "can convert compatible DynObj with context to LDObject" <| fun _ -> + let roc = Expect.wantSome (LDObject.tryFromDynamicObj compatibleDynObjWithContext) "LDObject.tryFromDynamicObj did not return Some" + Expect.equal roc all_properties_with_context "LDObject was not created correctly from compatible DynamicObj" + testCase "cannot convert incompatible DynObj to LDObject" <| fun _ -> + Expect.isNone (LDObject.tryFromDynamicObj incompatibleDynObj) "LDObject.tryFromDynamicObj did not return None" + ] ] ) From 45463bff0fdec1bd5e9aecd9b37bbfbe04eb5e0b Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Thu, 21 Nov 2024 17:04:31 +0100 Subject: [PATCH 2/3] LDObject: model `@type`/`additionalType` as arrays: - Json: parse `@type`/`additionalType` from either string (-> singleton) or array field, always write array - add tests --- src/Json/Decode.fs | 6 +- src/Json/LDObject.fs | 32 +++---- src/ROCrate/ArcROCrateMetadata.fs | 2 +- src/ROCrate/ISAProfile/Assay.fs | 2 +- src/ROCrate/ISAProfile/Data.fs | 6 +- src/ROCrate/ISAProfile/Dataset.fs | 10 +- src/ROCrate/ISAProfile/Investigation.fs | 2 +- src/ROCrate/ISAProfile/LabProcess.fs | 6 +- src/ROCrate/ISAProfile/LabProtocol.fs | 6 +- src/ROCrate/ISAProfile/Person.fs | 6 +- src/ROCrate/ISAProfile/PropertyValue.fs | 6 +- src/ROCrate/ISAProfile/Sample.fs | 6 +- src/ROCrate/ISAProfile/ScholarlyArticle.fs | 6 +- src/ROCrate/ISAProfile/Study.fs | 2 +- src/ROCrate/LDObject.fs | 20 ++-- tests/Json/LDObject.Tests.fs | 96 ++++++++++++++++--- tests/ROCrate/Common.fs | 31 ++++-- tests/ROCrate/ISAProfile/Assay.Tests.fs | 4 +- tests/ROCrate/ISAProfile/Data.Tests.fs | 6 +- tests/ROCrate/ISAProfile/Dataset.Tests.fs | 6 +- .../ROCrate/ISAProfile/Investigation.Tests.fs | 4 +- tests/ROCrate/ISAProfile/LabProcess.tests.fs | 6 +- tests/ROCrate/ISAProfile/LabProtocol.Tests.fs | 6 +- tests/ROCrate/ISAProfile/Person.Tests.fs | 6 +- .../ROCrate/ISAProfile/PropertyValue.Tests.fs | 6 +- tests/ROCrate/ISAProfile/Sample.tests.fs | 6 +- .../ISAProfile/ScholarlyArticle.Tests.fs | 6 +- tests/ROCrate/ISAProfile/Study.Tests.fs | 4 +- tests/ROCrate/LDObject.Tests.fs | 12 +-- .../TestingUtils/TestObjects.Json/ROCrate.fs | 81 +++++++++++++++- 30 files changed, 296 insertions(+), 102 deletions(-) diff --git a/src/Json/Decode.fs b/src/Json/Decode.fs index a52231f0..720991d0 100644 --- a/src/Json/Decode.fs +++ b/src/Json/Decode.fs @@ -93,9 +93,7 @@ module Decode = decoder.Decode(helpers,value) } - - - let resizeArray (decoder: Decoder<'value>) : Decoder> = + let resizeArrayOrSingleton (decoder: Decoder<'value>) : Decoder> = { new Decoder> with member _.Decode(helpers, value) = if helpers.isArray value then @@ -123,7 +121,7 @@ module Decode = Ok acc ) else - ("", BadPrimitive("an array", value)) |> Error + decoder.Decode(helpers, value) |> Result.map (fun x -> ResizeArray[x]) } let datetime: Decoder = diff --git a/src/Json/LDObject.fs b/src/Json/LDObject.fs index a9aae98c..6b5e1f34 100644 --- a/src/Json/LDObject.fs +++ b/src/Json/LDObject.fs @@ -7,7 +7,6 @@ open Thoth.Json.Core open DynamicObj module rec LDObject = - #if !FABLE_COMPILER let (|SomeObj|_|) = // create generalized option type @@ -45,21 +44,17 @@ module rec LDObject = | _ -> failwith "Unknown type" let rec encoder(obj: LDObject) = - obj.GetProperties true - |> Seq.choose (fun kv -> - let l = kv.Key.ToLower() - if l <> "id" && l <> "schematype" && l <> "additionaltype" then - Some(kv.Key, genericEncoder kv.Value) - else - None - - ) - |> Seq.append [ - "@id", Encode.string obj.Id - "@type", Encode.string obj.SchemaType - if obj.AdditionalType.IsSome then - "additionalType", Encode.string obj.AdditionalType.Value + + [ + yield "@id", Encode.string obj.Id |> Some + yield "@type", Encode.seq (obj.SchemaType |> Seq.map Encode.string) |> Some + yield Encode.tryIncludeSeq "additionalType" Encode.string obj.AdditionalType + for kv in (obj.GetProperties true) do + let l = kv.Key.ToLower() + if l <> "id" && l <> "schematype" && l <> "additionaltype" then + yield kv.Key, Some (genericEncoder kv.Value) ] + |> Encode.choose |> Encode.object /// Returns a decoder @@ -75,11 +70,12 @@ module rec LDObject = let properties = helpers.getProperties value let builder = fun (get : Decode.IGetters) -> - let t = get.Required.Field "@type" Decode.string + let t = get.Required.Field "@type" (Decode.resizeArrayOrSingleton Decode.string) let id = get.Required.Field "@id" Decode.string - let o = LDObject(id,t) + let at = get.Optional.Field "additionalType" (Decode.resizeArrayOrSingleton Decode.string) + let o = LDObject(id, t, ?additionalType = at) for property in properties do - if property <> "@id" && property <> "@type" then + if property <> "@id" && property <> "@type" && property <> "additionalType" then o.SetProperty(property,get.Required.Field property (decode(false))) o let result = builder getters diff --git a/src/ROCrate/ArcROCrateMetadata.fs b/src/ROCrate/ArcROCrateMetadata.fs index 2fd930b1..3e976b7e 100644 --- a/src/ROCrate/ArcROCrateMetadata.fs +++ b/src/ROCrate/ArcROCrateMetadata.fs @@ -4,7 +4,7 @@ open DynamicObj type ArcROCrateMetadata(?about : LDObject) as this = - inherit LDObject(id = "ro-crate-metadata",schemaType = "CreativeWork") + inherit LDObject(id = "ro-crate-metadata",schemaType = ResizeArray([|"CreativeWork"|])) do DynObj.setOptionalProperty (nameof about) about this diff --git a/src/ROCrate/ISAProfile/Assay.fs b/src/ROCrate/ISAProfile/Assay.fs index 5532ada4..1549799e 100644 --- a/src/ROCrate/ISAProfile/Assay.fs +++ b/src/ROCrate/ISAProfile/Assay.fs @@ -17,7 +17,7 @@ type Assay( ?url, ?variableMeasured ) as this = - inherit Dataset(id, "Assay") + inherit Dataset(id = id, additionalType = ResizeArray[|"Assay"|]) do DynObj.setProperty (nameof identifier) identifier this diff --git a/src/ROCrate/ISAProfile/Data.fs b/src/ROCrate/ISAProfile/Data.fs index 1f4e30ec..81797118 100644 --- a/src/ROCrate/ISAProfile/Data.fs +++ b/src/ROCrate/ISAProfile/Data.fs @@ -13,7 +13,11 @@ type Data( ?encodingFormat, ?disambiguatingDescription ) as this = - inherit LDObject(id = id, schemaType = "schema.org/MediaObject", ?additionalType = additionalType) + inherit LDObject( + id = id, + schemaType = ResizeArray[|"schema.org/MediaObject"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) do DynObj.setProperty (nameof name) name this diff --git a/src/ROCrate/ISAProfile/Dataset.fs b/src/ROCrate/ISAProfile/Dataset.fs index 52ce4779..c610b138 100644 --- a/src/ROCrate/ISAProfile/Dataset.fs +++ b/src/ROCrate/ISAProfile/Dataset.fs @@ -2,8 +2,12 @@ namespace ARCtrl.ROCrate open DynamicObj open Fable.Core - + /// [] -type Dataset (id: string, ?additionalType: string) = - inherit LDObject(id = id, schemaType = "schema.org/Dataset", ?additionalType = additionalType) +type Dataset (id: string, ?additionalType: ResizeArray) = + inherit LDObject( + id = id, + schemaType = ResizeArray[|"schema.org/Dataset"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) diff --git a/src/ROCrate/ISAProfile/Investigation.fs b/src/ROCrate/ISAProfile/Investigation.fs index 8e08aff7..0588caed 100644 --- a/src/ROCrate/ISAProfile/Investigation.fs +++ b/src/ROCrate/ISAProfile/Investigation.fs @@ -20,7 +20,7 @@ type Investigation( ?url, ?description ) as this = - inherit Dataset(id, "Investigation") + inherit Dataset(id = id, additionalType = ResizeArray[|"Investigation"|]) do DynObj.setProperty (nameof identifier) identifier this diff --git a/src/ROCrate/ISAProfile/LabProcess.fs b/src/ROCrate/ISAProfile/LabProcess.fs index 9abb4fdc..81a8648e 100644 --- a/src/ROCrate/ISAProfile/LabProcess.fs +++ b/src/ROCrate/ISAProfile/LabProcess.fs @@ -17,7 +17,11 @@ type LabProcess( ?endTime, ?disambiguatingDescription ) as this = - inherit LDObject(id = id, schemaType = "bioschemas.org/LabProcess", ?additionalType = additionalType) + inherit LDObject( + id = id, + schemaType = ResizeArray[|"bioschemas.org/LabProcess"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) do DynObj.setProperty (nameof name) name this DynObj.setProperty (nameof agent) agent this diff --git a/src/ROCrate/ISAProfile/LabProtocol.fs b/src/ROCrate/ISAProfile/LabProtocol.fs index db6a66d4..af8c2d9a 100644 --- a/src/ROCrate/ISAProfile/LabProtocol.fs +++ b/src/ROCrate/ISAProfile/LabProtocol.fs @@ -18,7 +18,11 @@ type LabProtocol( ?reagent, ?computationalTool ) as this = - inherit LDObject(id = id, schemaType = "bioschemas.org/LabProtocol", ?additionalType = additionalType) + inherit LDObject( + id = id, + schemaType = ResizeArray[|"bioschemas.org/LabProtocol"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) do DynObj.setOptionalProperty (nameof name) name this DynObj.setOptionalProperty (nameof intendedUse) intendedUse this diff --git a/src/ROCrate/ISAProfile/Person.fs b/src/ROCrate/ISAProfile/Person.fs index b3076813..01e8ea32 100644 --- a/src/ROCrate/ISAProfile/Person.fs +++ b/src/ROCrate/ISAProfile/Person.fs @@ -20,7 +20,11 @@ type Person( ?faxNumber, ?disambiguatingDescription ) as this= - inherit LDObject(id = id, schemaType = "schema.org/Person", ?additionalType = additionalType) + inherit LDObject( + id = id, + schemaType = ResizeArray[|"schema.org/Person"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) do DynObj.setProperty (nameof givenName) givenName this diff --git a/src/ROCrate/ISAProfile/PropertyValue.fs b/src/ROCrate/ISAProfile/PropertyValue.fs index 730ad7a7..b42f9ff7 100644 --- a/src/ROCrate/ISAProfile/PropertyValue.fs +++ b/src/ROCrate/ISAProfile/PropertyValue.fs @@ -15,7 +15,11 @@ type PropertyValue( ?valueReference, ?additionalType ) as this = - inherit LDObject(id = id, schemaType = "schema.org/PropertyValue", ?additionalType = additionalType) + inherit LDObject( + id = id, + schemaType = ResizeArray[|"schema.org/PropertyValue"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) do DynObj.setProperty (nameof name) name this diff --git a/src/ROCrate/ISAProfile/Sample.fs b/src/ROCrate/ISAProfile/Sample.fs index a607ce56..4604484a 100644 --- a/src/ROCrate/ISAProfile/Sample.fs +++ b/src/ROCrate/ISAProfile/Sample.fs @@ -12,7 +12,11 @@ type Sample( ?additionalProperty, ?derivesFrom ) as this = - inherit LDObject(id = id, schemaType = "bioschemas.org/Sample", ?additionalType = additionalType) + inherit LDObject( + id = id, + schemaType = ResizeArray[|"bioschemas.org/Sample"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) do DynObj.setProperty (nameof name) name this diff --git a/src/ROCrate/ISAProfile/ScholarlyArticle.fs b/src/ROCrate/ISAProfile/ScholarlyArticle.fs index cb4905a5..be1e00b8 100644 --- a/src/ROCrate/ISAProfile/ScholarlyArticle.fs +++ b/src/ROCrate/ISAProfile/ScholarlyArticle.fs @@ -16,7 +16,11 @@ type ScholarlyArticle( ?disambiguatingDescription ) as this = - inherit LDObject(id = id, schemaType = "schema.org/ScholarlyArticle", ?additionalType = additionalType) + inherit LDObject( + id = id, + schemaType = ResizeArray[|"schema.org/ScholarlyArticle"|], + additionalType = defaultArg additionalType (ResizeArray[||]) + ) do DynObj.setProperty (nameof headline) headline this diff --git a/src/ROCrate/ISAProfile/Study.fs b/src/ROCrate/ISAProfile/Study.fs index 41bbd82e..14134592 100644 --- a/src/ROCrate/ISAProfile/Study.fs +++ b/src/ROCrate/ISAProfile/Study.fs @@ -20,7 +20,7 @@ type Study( ?headline, ?url ) as this = - inherit Dataset(id, "Study") + inherit Dataset(id = id, additionalType = ResizeArray[|"Study"|]) do DynObj.setProperty (nameof identifier) identifier this DynObj.setOptionalProperty (nameof about) about this diff --git a/src/ROCrate/LDObject.fs b/src/ROCrate/LDObject.fs index 5dcc8745..e618e7a8 100644 --- a/src/ROCrate/LDObject.fs +++ b/src/ROCrate/LDObject.fs @@ -9,18 +9,18 @@ type LDContext() = inherit DynamicObj() /// Base interface implemented by all explicitly known objects in our ROCrate profiles. type ILDObject = - abstract member SchemaType : string with get, set + abstract member SchemaType : ResizeArray with get, set abstract member Id: string - abstract member AdditionalType: string option with get, set + abstract member AdditionalType: ResizeArray with get, set /// Base class for all explicitly known objects in our ROCrate profiles to inherit from. /// Basically a DynamicObj that implements the ILDObject interface. [] -type LDObject(id:string, schemaType: string, ?additionalType) = +type LDObject(id: string, schemaType: ResizeArray, ?additionalType: ResizeArray) = inherit DynamicObj() let mutable schemaType = schemaType - let mutable additionalType = additionalType + let mutable additionalType = defaultArg additionalType (ResizeArray []) member this.Id with get() = id @@ -60,15 +60,19 @@ type LDObject(id:string, schemaType: string, ?additionalType) = static member tryFromDynamicObj (dynObj: DynamicObj) = match - DynObj.tryGetTypedPropertyValue("@type") dynObj, + DynObj.tryGetTypedPropertyValue>("@type") dynObj, DynObj.tryGetTypedPropertyValue("@id") dynObj, - DynObj.tryGetTypedPropertyValue("additionalType") dynObj + DynObj.tryGetTypedPropertyValue>("additionalType") dynObj with - | (Some schemaType), (Some id), at -> - let roc = new LDObject(id, schemaType, ?additionalType = at) + | (Some st), (Some id), (Some at) -> + // initialize with extracted static members only + let roc = new LDObject(id = id, schemaType = st, additionalType = at) + + // copy dynamic properties! match DynObj.tryGetTypedPropertyValue("@context") dynObj with | Some context -> roc.SetContext(context) | _ -> () + dynObj.CopyDynamicPropertiesTo(roc) Some roc | _ -> None \ No newline at end of file diff --git a/tests/Json/LDObject.Tests.fs b/tests/Json/LDObject.Tests.fs index ad053dc7..8858a330 100644 --- a/tests/Json/LDObject.Tests.fs +++ b/tests/Json/LDObject.Tests.fs @@ -11,7 +11,7 @@ let private test_read = testList "Read" [ testCase "onlyIDAndType" <| fun _ -> let json = LDObject.fromROCrateJsonString(GenericObjects.onlyIDAndType) Expect.equal json.Id "MyIdentifier" "id was not parsed correctly" - Expect.equal json.SchemaType "MyType" "type was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" testCase "onlyID" <| fun _ -> let f = fun _ -> LDObject.fromROCrateJsonString(GenericObjects.onlyID) |> ignore Expect.throws f "Should fail if Type is missing" @@ -21,7 +21,7 @@ let private test_read = testList "Read" [ testCase "withStringFields" <| fun _ -> let json = LDObject.fromROCrateJsonString(GenericObjects.withStringFields) Expect.equal json.Id "MyIdentifier" "id was not parsed correctly" - Expect.equal json.SchemaType "MyType" "type was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" let name = Expect.wantSome (DynObj.tryGetTypedPropertyValue "name" json) "field name was not parsed" Expect.equal name "MyName" "field name was not parsed correctly" let description = Expect.wantSome (DynObj.tryGetTypedPropertyValue "description" json) "field description was not parsed" @@ -29,7 +29,7 @@ let private test_read = testList "Read" [ testCase "withIntFields" <| fun _ -> let json = LDObject.fromROCrateJsonString(GenericObjects.withIntFields) Expect.equal json.Id "MyIdentifier" "id was not parsed correctly" - Expect.equal json.SchemaType "MyType" "type was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" let number = Expect.wantSome (DynObj.tryGetTypedPropertyValue "number" json) "field number was not parsed" Expect.equal number 42 "field number was not parsed correctly" let anotherNumber = Expect.wantSome (DynObj.tryGetTypedPropertyValue "anotherNumber" json) "field anotherNumber was not parsed" @@ -37,7 +37,7 @@ let private test_read = testList "Read" [ testCase "withStringArray" <| fun _ -> let json = LDObject.fromROCrateJsonString(GenericObjects.withStringArray) Expect.equal json.Id "MyIdentifier" "id was not parsed correctly" - Expect.equal json.SchemaType "MyType" "type was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" let names = Expect.wantSome (DynObj.tryGetTypedPropertyValue> "names" json) "field names was not parsed" Expect.equal names.Count 2 "ResizeArray length is wrong" Expect.equal names.[0] "MyName" "First name was not parsed correctly" @@ -45,75 +45,147 @@ let private test_read = testList "Read" [ testCase "withNestedObject" <| fun _ -> let json = LDObject.fromROCrateJsonString(GenericObjects.withNestedObject) Expect.equal json.Id "OuterIdentifier" "id was not parsed correctly" - Expect.equal json.SchemaType "MyType" "type was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" let nested = Expect.wantSome (DynObj.tryGetTypedPropertyValue "nested" json) "field nested was not parsed" Expect.equal nested.Id "MyIdentifier" "nested id was not parsed correctly" - Expect.equal nested.SchemaType "MyType" "nested type was not parsed correctly" + Expect.sequenceEqual nested.SchemaType ResizeArray["MyType"] "nested type was not parsed correctly" testCase "withObjectArray" <| fun _ -> let json = LDObject.fromROCrateJsonString(GenericObjects.withObjectArray) Expect.equal json.Id "OuterIdentifier" "id was not parsed correctly" - Expect.equal json.SchemaType "MyType" "type was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" let nested = Expect.wantSome (DynObj.tryGetTypedPropertyValue> "nested" json) "field nested was not parsed" Expect.equal nested.Count 2 "ResizeArray length is wrong" let o1 = nested.[0] :?> LDObject Expect.equal o1.Id "MyIdentifier" "First nested id was not parsed correctly" - Expect.equal o1.SchemaType "MyType" "First nested type was not parsed correctly" + Expect.sequenceEqual o1.SchemaType ResizeArray["MyType"] "First nested type was not parsed correctly" let o2 = nested.[1] :?> LDObject Expect.equal o2.Id "MyIdentifier" "Second nested id was not parsed correctly" - Expect.equal o2.SchemaType "MyType" "Second nested type was not parsed correctly" + Expect.sequenceEqual o2.SchemaType ResizeArray["MyType"] "Second nested type was not parsed correctly" testCase "withMixedArray" <| fun _ -> let json = LDObject.fromROCrateJsonString(GenericObjects.withMixedArray) Expect.equal json.Id "OuterIdentifier" "id was not parsed correctly" - Expect.equal json.SchemaType "MyType" "type was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" let nested = Expect.wantSome (DynObj.tryGetTypedPropertyValue> "nested" json) "field nested was not parsed" Expect.equal nested.Count 3 "ResizeArray length is wrong" let o1 = nested.[0] :?> LDObject Expect.equal o1.Id "MyIdentifier" "First nested id of object was not parsed correctly" - Expect.equal o1.SchemaType "MyType" "First nested type of object was not parsed correctly" + Expect.sequenceEqual o1.SchemaType ResizeArray["MyType"] "First nested type of object was not parsed correctly" let o2 = nested.[1] :?> string Expect.equal o2 "Value2" "Second nested string was not parsed correctly" let o3 = nested.[2] :?> int Expect.equal o3 42 "Third nested int was not parsed correctly" + testCase "withAdditionalTypeString" <| fun _ -> + let json = LDObject.fromROCrateJsonString(GenericObjects.withAdditionalTypeString) + Expect.equal json.Id "MyIdentifier" "id was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" + Expect.sequenceEqual json.AdditionalType ResizeArray["additionalType"] "additionalType was not parsed correctly" + testCase "withAdditionalTypeArray" <| fun _ -> + let json = LDObject.fromROCrateJsonString(GenericObjects.withAdditionalTypeArray) + Expect.equal json.Id "MyIdentifier" "id was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" + Expect.sequenceEqual json.AdditionalType ResizeArray["additionalType"] "additionalType was not parsed correctly" + testCase "withAddtionalTypeArrayMultipleEntries" <| fun _ -> + let json = LDObject.fromROCrateJsonString(GenericObjects.withAddtionalTypeArrayMultipleEntries) + Expect.equal json.Id "MyIdentifier" "id was not parsed correctly" + Expect.sequenceEqual json.SchemaType ResizeArray["MyType"] "type was not parsed correctly" + Expect.sequenceEqual json.AdditionalType ResizeArray["additionalType1"; "additionalType2"] "additionalType was not parsed correctly" ] let test_write = testList "write" [ + // The tests suffixed with 'NoTypeArray' are not real roundtrips, as we parse string OR array fields but always write arrays for the @type field. testCase "onlyIDAndType" <| fun _ -> let json = GenericObjects.onlyIDAndType let object = LDObject.fromROCrateJsonString(json) let output = LDObject.toROCrateJsonString() object Expect.stringEqual output json "Output string is not correct" + testCase "onlyIDAndTypeNoTypeArray" <| fun _ -> + let json = GenericObjects.onlyIDAndTypeNoTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.onlyIDAndType "Output string is not correct" + testCase "withStringFields" <| fun _ -> let json = GenericObjects.withStringFields let object = LDObject.fromROCrateJsonString(json) let output = LDObject.toROCrateJsonString() object Expect.stringEqual output json "Output string is not correct" + testCase "withStringFieldsNoTypeArray" <| fun _ -> + let json = GenericObjects.withStringFieldsNoTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.withStringFields "Output string is not correct" + testCase "withIntFields" <| fun _ -> let json = GenericObjects.withIntFields let object = LDObject.fromROCrateJsonString(json) let output = LDObject.toROCrateJsonString() object Expect.stringEqual output json "Output string is not correct" + testCase "withIntFieldsNoTypeArray" <| fun _ -> + let json = GenericObjects.withIntFieldsNoTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.withIntFields "Output string is not correct" + testCase "withStringArray" <| fun _ -> let json = GenericObjects.withStringArray let object = LDObject.fromROCrateJsonString(json) let output = LDObject.toROCrateJsonString() object Expect.stringEqual output json "Output string is not correct" + testCase "withStringArrayNoTypeArray" <| fun _ -> + let json = GenericObjects.withStringArrayNoTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.withStringArray "Output string is not correct" + testCase "withNestedObject" <| fun _ -> let json = GenericObjects.withNestedObject let object = LDObject.fromROCrateJsonString(json) let output = LDObject.toROCrateJsonString() object Expect.stringEqual output json "Output string is not correct" + testCase "withNestedObjectNoTypeArray" <| fun _ -> + let json = GenericObjects.withNestedObjectNoTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.withNestedObject "Output string is not correct" + testCase "withObjectArray" <| fun _ -> let json = GenericObjects.withObjectArray let object = LDObject.fromROCrateJsonString(json) let output = LDObject.toROCrateJsonString() object Expect.stringEqual output json "Output string is not correct" + testCase "withObjectArrayNoTypeArray" <| fun _ -> + let json = GenericObjects.withObjectArrayNoTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.withObjectArray "Output string is not correct" + testCase "withMixedArray" <| fun _ -> let json = GenericObjects.withMixedArray let object = LDObject.fromROCrateJsonString(json) let output = LDObject.toROCrateJsonString() object Expect.stringEqual output json "Output string is not correct" - + testCase "withMixedArrayNoTypeArray" <| fun _ -> + let json = GenericObjects.withMixedArrayNoTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.withMixedArray "Output string is not correct" + + testCase "withAddtionalTypeArray" <| fun _ -> + let json = GenericObjects.withAdditionalTypeArray + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output json "Output string is not correct" + testCase "withAddtionalTypeArrayMultipleEntries" <| fun _ -> + let json = GenericObjects.withAddtionalTypeArrayMultipleEntries + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output json "Output string is not correct" + testCase "withAddtionalTypeString" <| fun _ -> + let json = GenericObjects.withAdditionalTypeString + let object = LDObject.fromROCrateJsonString(json) + let output = LDObject.toROCrateJsonString() object + Expect.stringEqual output GenericObjects.withAdditionalTypeArray "Output string is not correct" ] let main = testList "LDObject" [ diff --git a/tests/ROCrate/Common.fs b/tests/ROCrate/Common.fs index 0c56cc5f..2a89cb2c 100644 --- a/tests/ROCrate/Common.fs +++ b/tests/ROCrate/Common.fs @@ -11,11 +11,28 @@ module Expect = Expect.equal roc.Id expectedId "object did not contain correct @id" let inline LDObjectHasType (expectedType:string) (roc:#LDObject) = - Expect.equal roc.SchemaType expectedType "object did not contain correct @type" + Expect.containsAll + roc.SchemaType + [expectedType] + "object did not contain correct @type" + + let inline LDObjectHasTypes (expectedTypes:seq) (roc:#LDObject) = + Expect.containsAll + roc.SchemaType + expectedTypes + "object did not contain correct @types" let inline LDObjectHasAdditionalType (expectedAdditionalType:string) (roc:#LDObject) = - Expect.isSome roc.AdditionalType "additionalType was None" - Expect.equal roc.AdditionalType (Some expectedAdditionalType) "object did not contain correct additionalType" + Expect.containsAll + roc.AdditionalType + [expectedAdditionalType] + "object did not contain correct additionalType" + + let inline LDObjectHasAdditionalTypes (expectedAdditionalTypes:seq) (roc:#LDObject) = + Expect.containsAll + roc.AdditionalType + expectedAdditionalTypes + "object did not contain correct additionalTypes" let inline LDObjectHasDynamicProperty (expectedPropertyName:string) (expectedPropertyValue:'P) (roc:#LDObject) = Expect.isSome (roc.TryGetDynamicPropertyHelper(expectedPropertyName)) $"object did not contain the dynamic property '{expectedPropertyName}'" @@ -25,14 +42,14 @@ module Expect = $"property value of '{expectedPropertyName}' was not correct" let inline LDObjectHasStaticProperty (expectedPropertyName:string) (expectedPropertyValue:'P) (roc:#LDObject) = - Expect.isSome (roc.TryGetDynamicPropertyHelper(expectedPropertyName)) $"object did not contain the dynamic property '{expectedPropertyName}'" + Expect.isSome (roc.TryGetStaticPropertyHelper(expectedPropertyName)) $"object did not contain the static property '{expectedPropertyName}'" Expect.equal (DynObj.tryGetTypedPropertyValue<'P> expectedPropertyName roc) (Some expectedPropertyValue) $"property value of '{expectedPropertyName}' was not correct" - let inline LDObjectHasExpectedInterfaceMembers (expectedType:string) (expectedId:string) (expectedAdditionalType:string option) (roc:#LDObject) = + let inline LDObjectHasExpectedInterfaceMembers (expectedTypes: seq) (expectedId:string) (expectedAdditionalTypes: seq) (roc:#LDObject) = let interfacerino = roc :> ILDObject - Expect.equal interfacerino.SchemaType expectedType "object did not contain correct @type via interface access" + Expect.sequenceEqual interfacerino.SchemaType expectedTypes "object did not contain correct @types via interface access" Expect.equal interfacerino.Id expectedId "object did not contain correct @id via interface access" - Expect.equal interfacerino.AdditionalType expectedAdditionalType "object did not contain correct additionalType via interface access" + Expect.sequenceEqual interfacerino.AdditionalType expectedAdditionalTypes "object did not contain correct additionalTypes via interface access" diff --git a/tests/ROCrate/ISAProfile/Assay.Tests.fs b/tests/ROCrate/ISAProfile/Assay.Tests.fs index 70eb7a86..5d60761d 100644 --- a/tests/ROCrate/ISAProfile/Assay.Tests.fs +++ b/tests/ROCrate/ISAProfile/Assay.Tests.fs @@ -48,8 +48,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "assay_mandatory_properties_id" (Some "Assay") mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "assay_all_properties_id" (Some "Assay") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "assay_mandatory_properties_id" [|"Assay"|] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "assay_all_properties_id" [|"Assay"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/Data.Tests.fs b/tests/ROCrate/ISAProfile/Data.Tests.fs index bbeb6716..45fae9f3 100644 --- a/tests/ROCrate/ISAProfile/Data.Tests.fs +++ b/tests/ROCrate/ISAProfile/Data.Tests.fs @@ -14,7 +14,7 @@ let mandatory_properties = Data( let all_properties = Data( id = "data_all_properties_id", name = "name", - additionalType = "additionalType", + additionalType = ResizeArray([|"additionalType"|]), comment = "comment", encodingFormat = "encodingFormat", disambiguatingDescription = "disambiguatingDescription" @@ -38,8 +38,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/MediaObject" "data_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/MediaObject" "data_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/MediaObject"|] "data_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/MediaObject"|] "data_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/Dataset.Tests.fs b/tests/ROCrate/ISAProfile/Dataset.Tests.fs index f4468679..519a7aee 100644 --- a/tests/ROCrate/ISAProfile/Dataset.Tests.fs +++ b/tests/ROCrate/ISAProfile/Dataset.Tests.fs @@ -7,7 +7,7 @@ open TestingUtils open Common let mandatory_properties = Dataset("dataset_mandatory_properties_id") -let all_properties = Dataset("dataset_all_properties_id", additionalType = "additionalType") +let all_properties = Dataset("dataset_all_properties_id", additionalType = ResizeArray([|"additionalType"|])) let tests_profile_object_is_valid = testList "constructed properties" [ testList "mandatory properties" [ @@ -22,8 +22,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "dataset_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "dataset_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "dataset_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "dataset_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/Investigation.Tests.fs b/tests/ROCrate/ISAProfile/Investigation.Tests.fs index 8e051460..b42f6c65 100644 --- a/tests/ROCrate/ISAProfile/Investigation.Tests.fs +++ b/tests/ROCrate/ISAProfile/Investigation.Tests.fs @@ -54,8 +54,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "investigation_mandatory_properties_id" (Some "Investigation") mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "investigation_all_properties_id" (Some "Investigation") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "investigation_mandatory_properties_id" [|"Investigation"|] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "investigation_all_properties_id" [|"Investigation"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/LabProcess.tests.fs b/tests/ROCrate/ISAProfile/LabProcess.tests.fs index 23971baf..6732ea5c 100644 --- a/tests/ROCrate/ISAProfile/LabProcess.tests.fs +++ b/tests/ROCrate/ISAProfile/LabProcess.tests.fs @@ -20,7 +20,7 @@ let all_properties = LabProcess( agent = "agent", object = "object", result = "result", - additionalType = "additionalType", + additionalType = ResizeArray([|"additionalType"|]), executesLabProtocol = "executesLabProtocol", parameterValue = "parameterValue", endTime = "endTime", @@ -53,8 +53,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "bioschemas.org/LabProcess" "labprocess_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "bioschemas.org/LabProcess" "labprocess_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"bioschemas.org/LabProcess"|] "labprocess_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"bioschemas.org/LabProcess"|] "labprocess_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/LabProtocol.Tests.fs b/tests/ROCrate/ISAProfile/LabProtocol.Tests.fs index 588036c0..1c88e3eb 100644 --- a/tests/ROCrate/ISAProfile/LabProtocol.Tests.fs +++ b/tests/ROCrate/ISAProfile/LabProtocol.Tests.fs @@ -12,7 +12,7 @@ let mandatory_properties = LabProtocol( let all_properties = LabProtocol( id = "labprotocol_all_properties_id", - additionalType = "additionalType", + additionalType = ResizeArray([|"additionalType"|]), name = "name", intendedUse = "intendedUse", description = "description", @@ -46,8 +46,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "bioschemas.org/LabProtocol" "labprotocol_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "bioschemas.org/LabProtocol" "labprotocol_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"bioschemas.org/LabProtocol"|] "labprotocol_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"bioschemas.org/LabProtocol"|] "labprotocol_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/Person.Tests.fs b/tests/ROCrate/ISAProfile/Person.Tests.fs index 107dfb28..70bc372e 100644 --- a/tests/ROCrate/ISAProfile/Person.Tests.fs +++ b/tests/ROCrate/ISAProfile/Person.Tests.fs @@ -14,7 +14,7 @@ let mandatory_properties = Person( let all_properties = Person( id = "person_all_properties_id", givenName = "givenName", - additionalType = "additionalType", + additionalType = ResizeArray([|"additionalType"|]), familyName = "familyName", email = "email", identifier = "identifier", @@ -52,8 +52,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Person" "person_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Person" "person_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Person"|] "person_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Person"|] "person_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/PropertyValue.Tests.fs b/tests/ROCrate/ISAProfile/PropertyValue.Tests.fs index c44ec118..016ca86c 100644 --- a/tests/ROCrate/ISAProfile/PropertyValue.Tests.fs +++ b/tests/ROCrate/ISAProfile/PropertyValue.Tests.fs @@ -20,7 +20,7 @@ let all_properties = PropertyValue( unitCode = "unitCode", unitText = "unitText", valueReference = "valueReference", - additionalType = "additionalType" + additionalType = ResizeArray([|"additionalType"|]) ) let tests_profile_object_is_valid = testList "constructed properties" [ @@ -44,8 +44,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/PropertyValue" "propertyvalue_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/PropertyValue" "propertyvalue_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/PropertyValue"|] "propertyvalue_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/PropertyValue"|] "propertyvalue_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/Sample.tests.fs b/tests/ROCrate/ISAProfile/Sample.tests.fs index 659cb47d..d5f26f31 100644 --- a/tests/ROCrate/ISAProfile/Sample.tests.fs +++ b/tests/ROCrate/ISAProfile/Sample.tests.fs @@ -14,7 +14,7 @@ let mandatory_properties = Sample( let all_properties = Sample( id = "sample_all_properties_id", name = "name", - additionalType = "additionalType", + additionalType = ResizeArray([|"additionalType"|]), additionalProperty = "additionalProperty", derivesFrom = "derivesFrom" ) @@ -36,8 +36,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "bioschemas.org/Sample" "sample_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "bioschemas.org/Sample" "sample_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"bioschemas.org/Sample"|] "sample_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"bioschemas.org/Sample"|] "sample_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/ScholarlyArticle.Tests.fs b/tests/ROCrate/ISAProfile/ScholarlyArticle.Tests.fs index ea2c1168..13a1d12f 100644 --- a/tests/ROCrate/ISAProfile/ScholarlyArticle.Tests.fs +++ b/tests/ROCrate/ISAProfile/ScholarlyArticle.Tests.fs @@ -16,7 +16,7 @@ let all_properties = ScholarlyArticle( id = "scholarlyarticle_all_properties_id", headline = "headline", identifier = "identifier", - additionalType = "additionalType", + additionalType = ResizeArray([|"additionalType"|]), author = "author", url = "url", creativeWorkStatus = "creativeWorkStatus", @@ -44,8 +44,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/ScholarlyArticle" "scholarlyarticle_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/ScholarlyArticle" "scholarlyarticle_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/ScholarlyArticle"|] "scholarlyarticle_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/ScholarlyArticle"|] "scholarlyarticle_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/ISAProfile/Study.Tests.fs b/tests/ROCrate/ISAProfile/Study.Tests.fs index b987aef2..352b445b 100644 --- a/tests/ROCrate/ISAProfile/Study.Tests.fs +++ b/tests/ROCrate/ISAProfile/Study.Tests.fs @@ -54,8 +54,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "study_mandatory_properties_id" (Some "Study") mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "schema.org/Dataset" "study_all_properties_id" (Some "Study") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "study_mandatory_properties_id" [|"Study"|] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"schema.org/Dataset"|] "study_all_properties_id" [|"Study"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/ROCrate/LDObject.Tests.fs b/tests/ROCrate/LDObject.Tests.fs index 98b3b9ea..0b6db2e6 100644 --- a/tests/ROCrate/LDObject.Tests.fs +++ b/tests/ROCrate/LDObject.Tests.fs @@ -10,14 +10,14 @@ let context = new LDContext() |> DynObj.withProperty "more" "context" -let mandatory_properties = LDObject("LDObject_mandatory_properties_id", "someType") +let mandatory_properties = LDObject("LDObject_mandatory_properties_id", ResizeArray[|"someType"|]) let mandatory_properties_with_context = - LDObject("LDObject_mandatory_properties_id", "someType") + LDObject("LDObject_mandatory_properties_id", ResizeArray[|"someType"|]) |> DynObj.withProperty "@context" context -let all_properties = LDObject("LDObject_all_properties_id", "someType", additionalType = "additionalType") +let all_properties = LDObject("LDObject_all_properties_id", ResizeArray[|"someType"|], additionalType = ResizeArray[|"additionalType"|]) let all_properties_with_context = - LDObject("LDObject_all_properties_id", "someType", additionalType = "additionalType") + LDObject("LDObject_all_properties_id", ResizeArray[|"someType"|], additionalType = ResizeArray[|"additionalType"|]) |> DynObj.withProperty "@context" (context.CopyDynamicProperties()) let tests_profile_object_is_valid = testList "constructed properties" [ @@ -33,8 +33,8 @@ let tests_profile_object_is_valid = testList "constructed properties" [ ] let tests_interface_members = testList "interface members" [ - testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "someType" "LDObject_mandatory_properties_id" None mandatory_properties - testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers "someType" "LDObject_all_properties_id" (Some "additionalType") all_properties + testCase "mandatoryProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"someType"|] "LDObject_mandatory_properties_id" [||] mandatory_properties + testCase "allProperties" <| fun _ -> Expect.LDObjectHasExpectedInterfaceMembers [|"someType"|] "LDObject_all_properties_id" [|"additionalType"|] all_properties ] let tests_dynamic_members = testSequenced ( diff --git a/tests/TestingUtils/TestObjects.Json/ROCrate.fs b/tests/TestingUtils/TestObjects.Json/ROCrate.fs index 4d403ca6..0087e750 100644 --- a/tests/TestingUtils/TestObjects.Json/ROCrate.fs +++ b/tests/TestingUtils/TestObjects.Json/ROCrate.fs @@ -204,6 +204,12 @@ let publication = """{ module GenericObjects = let onlyIDAndType = + """{ + "@id": "MyIdentifier", + "@type": ["MyType"] + }""" + + let onlyIDAndTypeNoTypeArray = """{ "@id": "MyIdentifier", "@type": "MyType" @@ -215,11 +221,24 @@ module GenericObjects = }""" let onlyType = + """{ + "@type": ["MyType"] + }""" + + let onlyTypeNoTypeArray = """{ "@type": "MyType" }""" let withStringFields = + """{ + "@id": "MyIdentifier", + "@type": ["MyType"], + "name": "MyName", + "description": "MyDescription" + }""" + + let withStringFieldsNoTypeArray = """{ "@id": "MyIdentifier", "@type": "MyType", @@ -228,6 +247,14 @@ module GenericObjects = }""" let withIntFields = + """{ + "@id": "MyIdentifier", + "@type": ["MyType"], + "number": 42, + "anotherNumber": 1337 + }""" + + let withIntFieldsNoTypeArray = """{ "@id": "MyIdentifier", "@type": "MyType", @@ -236,6 +263,13 @@ module GenericObjects = }""" let withStringArray = + """{ + "@id": "MyIdentifier", + "@type": ["MyType"], + "names": ["MyName", "MySecondName"] + }""" + + let withStringArrayNoTypeArray = """{ "@id": "MyIdentifier", "@type": "MyType", @@ -245,20 +279,61 @@ module GenericObjects = let withNestedObject = sprintf """{ "@id": "OuterIdentifier", - "@type": "MyType", + "@type": ["MyType"], "nested": %s }""" onlyIDAndType - let withObjectArray = + let withNestedObjectNoTypeArray = sprintf """{ "@id": "OuterIdentifier", "@type": "MyType", + "nested": %s + }""" onlyIDAndTypeNoTypeArray + + let withObjectArray = + sprintf """{ + "@id": "OuterIdentifier", + "@type": ["MyType"], "nested": [%s, %s] }""" onlyIDAndType onlyIDAndType + let withObjectArrayNoTypeArray = + sprintf """{ + "@id": "OuterIdentifier", + "@type": "MyType", + "nested": [%s, %s] + }""" onlyIDAndTypeNoTypeArray onlyIDAndTypeNoTypeArray + let withMixedArray = + sprintf """{ + "@id": "OuterIdentifier", + "@type": ["MyType"], + "nested": [%s, "Value2", 42] + }""" onlyIDAndType + + let withMixedArrayNoTypeArray = sprintf """{ "@id": "OuterIdentifier", "@type": "MyType", "nested": [%s, "Value2", 42] - }""" onlyIDAndType \ No newline at end of file + }""" onlyIDAndTypeNoTypeArray + + let withAdditionalTypeString = + """{ + "@id": "MyIdentifier", + "@type": ["MyType"], + "additionalType": "additionalType" + }""" + + let withAdditionalTypeArray = + """{ + "@id": "MyIdentifier", + "@type": ["MyType"], + "additionalType": ["additionalType"] + }""" + let withAddtionalTypeArrayMultipleEntries = + """{ + "@id": "MyIdentifier", + "@type": ["MyType"], + "additionalType": ["additionalType1", "additionalType2"] + }""" \ No newline at end of file From f5933759d74481296dc5aaf01eeffeaee176761c Mon Sep 17 00:00:00 2001 From: Kevin Schneider Date: Thu, 19 Dec 2024 10:40:45 +0100 Subject: [PATCH 3/3] Fix running single test projects --- build/ProjectInfo.fs | 21 ++++++++++++--------- build/TestTasks.fs | 35 ++++++++++++++++------------------- 2 files changed, 28 insertions(+), 28 deletions(-) diff --git a/build/ProjectInfo.fs b/build/ProjectInfo.fs index cf0aa003..976bc619 100644 --- a/build/ProjectInfo.fs +++ b/build/ProjectInfo.fs @@ -5,19 +5,22 @@ open Helpers let project = "ARCtrl" +let allTestsProject = "tests/All" + /// Dotnet and JS test paths let testProjects = [ "tests/All" - //"tests/Core" - //"tests/Json" - //"tests/Spreadsheet" - //"tests/FileSystem" - //"tests/ARCtrl" - //"tests/Yaml" - //"tests/ValidationPackages" - //"tests/Contract" - //"tests/ROCrate" + "tests/ARCtrl" + "tests/Contract" + "tests/Core" + "tests/CWL" + "tests/FileSystem" + "tests/Json" + "tests/ROCrate" + "tests/Spreadsheet" + "tests/ValidationPackages" + "tests/Yaml" ] /// Native JS test paths diff --git a/build/TestTasks.fs b/build/TestTasks.fs index 6f4eb765..e6654997 100644 --- a/build/TestTasks.fs +++ b/build/TestTasks.fs @@ -36,16 +36,14 @@ module RunTests = let runTestsJs = BuildTask.createFn "runTestsJS" [clean] (fun tp -> if tp.Context.Arguments |> List.exists (fun a -> a.ToLower() = skipTestsFlag.ToLower()) |> not then Trace.traceImportant "Start Js tests" - for path in ProjectInfo.testProjects do - // Setup test results directory after clean - System.IO.Directory.CreateDirectory("./tests/TestingUtils/TestResults/js") |> ignore - // transpile js files from fsharp code - run dotnet $"fable {path} -o {path}/js --nocache" "" - - System.IO.File.Copy(jsHelperFilePath, $"{path}/js/{jsHelperFileName}") |> ignore - // run mocha in target path to execute tests - // "--timeout 20000" is used, because json schema validation takes a bit of time. - run node $"{path}/js/Main.js" "" + // Setup test results directory after clean + System.IO.Directory.CreateDirectory("./tests/TestingUtils/TestResults/js") |> ignore + // transpile js files from fsharp code + run dotnet $"fable {allTestsProject} -o {allTestsProject}/js --nocache" "" + System.IO.File.Copy(jsHelperFilePath, $"{allTestsProject}/js/{jsHelperFileName}") |> ignore + // run mocha in target path to execute tests + // "--timeout 20000" is used, because json schema validation takes a bit of time. + run node $"{allTestsProject}/js/Main.js" "" else Trace.traceImportant "Skipping Js tests" ) @@ -65,13 +63,12 @@ module RunTests = let runTestsPy = BuildTask.createFn "runTestsPy" [clean] (fun tp -> if tp.Context.Arguments |> List.exists (fun a -> a.ToLower() = skipTestsFlag.ToLower()) |> not then Trace.traceImportant "Start Python tests" - for path in ProjectInfo.testProjects do - // Setup test results directory after clean - System.IO.Directory.CreateDirectory("./tests/TestingUtils/TestResults/py") |> ignore - //transpile py files from fsharp code - run dotnet $"fable {path} -o {path}/py --lang python --nocache" "" - // run pyxpecto in target path to execute tests in python - run python $"{path}/py/main.py" "" + // Setup test results directory after clean + System.IO.Directory.CreateDirectory("./tests/TestingUtils/TestResults/py") |> ignore + //transpile py files from fsharp code + run dotnet $"fable {allTestsProject} -o {allTestsProject}/py --lang python --nocache" "" + // run pyxpecto in target path to execute tests in python + run python $"{allTestsProject}/py/main.py" "" else Trace.traceImportant "Skipping Python tests" @@ -81,8 +78,7 @@ module RunTests = if tp.Context.Arguments |> List.exists (fun a -> a.ToLower() = skipTestsFlag.ToLower()) |> not then Trace.traceImportant "Start .NET tests" let dotnetRun = run dotnet "run" - testProjects - |> Seq.iter dotnetRun + dotnetRun allTestsProject else Trace.traceImportant "Skipping .NET tests" ) @@ -105,6 +101,7 @@ module RunTests = run python $"{p}/py/main.py" "" // transpile js files from fsharp code run dotnet $"fable {p} -o {p}/js" "" + System.IO.Directory.CreateDirectory("./tests/TestingUtils/TestResults/js") |> ignore System.IO.File.Copy(jsHelperFilePath, $"{p}/js/{jsHelperFileName}") |> ignore // run mocha in target path to execute tests // "--timeout 20000" is used, because json schema validation takes a bit of time.