mirror of
				https://github.com/Smaug123/WoofWare.Myriad
				synced 2025-10-25 22:08:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			672 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			Forth
		
	
	
	
	
	
			
		
		
	
	
			672 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			Forth
		
	
	
	
	
	
| namespace WoofWare.Myriad.Plugins
 | |
| 
 | |
| open System
 | |
| open System.Text
 | |
| open Fantomas.FCS.Syntax
 | |
| open Fantomas.FCS.SyntaxTrivia
 | |
| open Fantomas.FCS.Xml
 | |
| open Myriad.Core
 | |
| 
 | |
| /// Attribute indicating a record type to which the "Add JSON parse" Myriad
 | |
| /// generator should apply during build.
 | |
| /// The purpose of this generator is to create methods (possibly extension methods) of the form
 | |
| /// `{TypeName}.jsonParse : System.Text.Json.Nodes.JsonNode -> {TypeName}`.
 | |
| ///
 | |
| /// If you supply isExtensionMethod = true, you will get extension methods.
 | |
| /// These can only be consumed from F#, but the benefit is that they don't use up the module name
 | |
| /// (since by default we create a module called "{TypeName}").
 | |
| type JsonParseAttribute (isExtensionMethod : bool) =
 | |
|     inherit Attribute ()
 | |
| 
 | |
|     /// If changing this, *adjust the documentation strings*
 | |
|     static member internal DefaultIsExtensionMethod = false
 | |
| 
 | |
|     /// Shorthand for the "isExtensionMethod = false" constructor; see documentation there for details.
 | |
|     new () = JsonParseAttribute JsonParseAttribute.DefaultIsExtensionMethod
 | |
| 
 | |
| type internal JsonParseOutputSpec =
 | |
|     {
 | |
|         ExtensionMethods : bool
 | |
|     }
 | |
| 
 | |
| [<RequireQualifiedAccess>]
 | |
| module internal JsonParseGenerator =
 | |
|     open Fantomas.FCS.Text.Range
 | |
|     open Myriad.Core.Ast
 | |
| 
 | |
|     type JsonParseOption =
 | |
|         {
 | |
|             JsonNumberHandlingArg : SynExpr option
 | |
|         }
 | |
| 
 | |
|         static member None =
 | |
|             {
 | |
|                 JsonNumberHandlingArg = None
 | |
|             }
 | |
| 
 | |
|     /// (match {indexed} with | null -> raise (System.Collections.Generic.KeyNotFoundException ()) | v -> v)
 | |
|     let assertNotNull (propertyName : SynExpr) (indexed : SynExpr) =
 | |
|         let raiseExpr =
 | |
|             SynExpr.CreateApp (
 | |
|                 SynExpr.CreateIdentString "raise",
 | |
|                 SynExpr.CreateParen (
 | |
|                     SynExpr.CreateApp (
 | |
|                         SynExpr.CreateLongIdent (
 | |
|                             SynLongIdent.Create [ "System" ; "Collections" ; "Generic" ; "KeyNotFoundException" ]
 | |
|                         ),
 | |
|                         SynExpr.CreateParen (
 | |
|                             SynExpr.CreateApp (
 | |
|                                 SynExpr.CreateApp (
 | |
|                                     SynExpr.CreateIdentString "sprintf",
 | |
|                                     SynExpr.CreateConstString "Required key '%s' not found on JSON object"
 | |
|                                 ),
 | |
|                                 SynExpr.CreateParen propertyName
 | |
|                             )
 | |
|                         )
 | |
|                     )
 | |
|                 )
 | |
|             )
 | |
| 
 | |
|         SynExpr.CreateMatch (
 | |
|             indexed,
 | |
|             [
 | |
|                 SynMatchClause.Create (SynPat.CreateNull, None, raiseExpr)
 | |
|                 SynMatchClause.Create (SynPat.CreateNamed (Ident.Create "v"), None, SynExpr.CreateIdentString "v")
 | |
|             ]
 | |
|         )
 | |
|         |> SynExpr.CreateParen
 | |
| 
 | |
|     /// {node}.AsValue().GetValue<{typeName}> ()
 | |
|     /// If `propertyName` is Some, uses `assertNotNull {node}` instead of `{node}`.
 | |
|     let asValueGetValue (propertyName : SynExpr option) (typeName : string) (node : SynExpr) : SynExpr =
 | |
|         match propertyName with
 | |
|         | None -> node
 | |
|         | Some propertyName -> assertNotNull propertyName node
 | |
|         |> SynExpr.callMethod "AsValue"
 | |
|         |> SynExpr.callGenericMethod "GetValue" typeName
 | |
| 
 | |
|     /// {node}.AsObject()
 | |
|     /// If `propertyName` is Some, uses `assertNotNull {node}` instead of `{node}`.
 | |
|     let asObject (propertyName : SynExpr option) (node : SynExpr) : SynExpr =
 | |
|         match propertyName with
 | |
|         | None -> node
 | |
|         | Some propertyName -> assertNotNull propertyName node
 | |
|         |> SynExpr.callMethod "AsObject"
 | |
| 
 | |
|     /// {type}.jsonParse {node}
 | |
|     let typeJsonParse (typeName : LongIdent) (node : SynExpr) : SynExpr =
 | |
|         SynExpr.CreateApp (
 | |
|             SynExpr.CreateLongIdent (SynLongIdent.CreateFromLongIdent (typeName @ [ Ident.Create "jsonParse" ])),
 | |
|             node
 | |
|         )
 | |
| 
 | |
|     /// collectionType is e.g. "List"; we'll be calling `ofSeq` on it.
 | |
|     /// body is the body of a lambda which takes a parameter `elt`.
 | |
|     /// {assertNotNull node}.AsArray()
 | |
|     /// |> Seq.map (fun elt -> {body})
 | |
|     /// |> {collectionType}.ofSeq
 | |
|     let asArrayMapped
 | |
|         (propertyName : SynExpr option)
 | |
|         (collectionType : string)
 | |
|         (node : SynExpr)
 | |
|         (body : SynExpr)
 | |
|         : SynExpr
 | |
|         =
 | |
|         match propertyName with
 | |
|         | None -> node
 | |
|         | Some propertyName -> assertNotNull propertyName node
 | |
|         |> SynExpr.callMethod "AsArray"
 | |
|         |> SynExpr.pipeThroughFunction (
 | |
|             SynExpr.CreateApp (
 | |
|                 SynExpr.CreateLongIdent (SynLongIdent.Create [ "Seq" ; "map" ]),
 | |
|                 SynExpr.createLambda "elt" body
 | |
|             )
 | |
|         )
 | |
|         |> SynExpr.pipeThroughFunction (SynExpr.CreateLongIdent (SynLongIdent.Create [ collectionType ; "ofSeq" ]))
 | |
| 
 | |
|     /// match {node} with | null -> None | v -> {body} |> Some
 | |
|     /// Use the variable `v` to get access to the `Some`.
 | |
|     let createParseLineOption (node : SynExpr) (body : SynExpr) : SynExpr =
 | |
|         let body = SynExpr.pipeThroughFunction (SynExpr.CreateIdentString "Some") body
 | |
| 
 | |
|         SynExpr.CreateMatch (
 | |
|             node,
 | |
|             [
 | |
|                 SynMatchClause.Create (SynPat.CreateNull, None, SynExpr.CreateIdent (Ident.Create "None"))
 | |
|                 SynMatchClause.Create (SynPat.CreateNamed (Ident.Create "v"), None, body)
 | |
|             ]
 | |
|         )
 | |
| 
 | |
|     /// Given e.g. "float", returns "System.Double.Parse"
 | |
|     let parseFunction (typeName : string) : LongIdent =
 | |
|         List.append (SynExpr.qualifyPrimitiveType typeName) [ Ident.Create "Parse" ]
 | |
| 
 | |
|     /// fun kvp -> let key = {key(kvp)} in let value = {value(kvp)} in (key, value))
 | |
|     /// The inputs will be fed with appropriate SynExprs to apply them to the `kvp.Key` and `kvp.Value` args.
 | |
|     let dictionaryMapper (key : SynExpr -> SynExpr) (value : SynExpr -> SynExpr) : SynExpr =
 | |
|         let keyArg =
 | |
|             SynExpr.CreateLongIdent (SynLongIdent.Create [ "kvp" ; "Key" ])
 | |
|             |> SynExpr.CreateParen
 | |
| 
 | |
|         let valueArg =
 | |
|             SynExpr.CreateLongIdent (SynLongIdent.Create [ "kvp" ; "Value" ])
 | |
|             |> SynExpr.CreateParen
 | |
| 
 | |
|         SynExpr.LetOrUse (
 | |
|             false,
 | |
|             false,
 | |
|             [
 | |
|                 SynBinding.Let (pattern = SynPat.CreateNamed (Ident.Create "key"), expr = key keyArg)
 | |
|             ],
 | |
|             SynExpr.LetOrUse (
 | |
|                 false,
 | |
|                 false,
 | |
|                 [
 | |
|                     SynBinding.Let (pattern = SynPat.CreateNamed (Ident.Create "value"), expr = value valueArg)
 | |
|                 ],
 | |
|                 SynExpr.CreateTuple [ SynExpr.CreateIdentString "key" ; SynExpr.CreateIdentString "value" ],
 | |
|                 range0,
 | |
|                 {
 | |
|                     InKeyword = None
 | |
|                 }
 | |
|             ),
 | |
|             range0,
 | |
|             {
 | |
|                 InKeyword = None
 | |
|             }
 | |
|         )
 | |
|         |> SynExpr.createLambda "kvp"
 | |
| 
 | |
|     /// A conforming JSON object has only strings as keys. But it would be reasonable to allow the user
 | |
|     /// to parse these as URIs, for example.
 | |
|     let parseKeyString (desiredType : SynType) (key : SynExpr) : SynExpr =
 | |
|         match desiredType with
 | |
|         | String -> key
 | |
|         | Uri ->
 | |
|             key
 | |
|             |> SynExpr.pipeThroughFunction (SynExpr.CreateLongIdent (SynLongIdent.Create [ "System" ; "Uri" ]))
 | |
|         | _ ->
 | |
|             failwithf
 | |
|                 $"Unable to parse the key type %+A{desiredType} of a JSON object. Keys are strings, and this plugin does not know how to convert to that from a string."
 | |
| 
 | |
|     /// Given `node.["town"]`, for example, choose how to obtain a JSON value from it.
 | |
|     /// The property name is used in error messages at runtime to show where a JSON
 | |
|     /// parse error occurred; supply `None` to indicate "don't validate".
 | |
|     let rec parseNode
 | |
|         (propertyName : SynExpr option)
 | |
|         (options : JsonParseOption)
 | |
|         (fieldType : SynType)
 | |
|         (node : SynExpr)
 | |
|         : SynExpr
 | |
|         =
 | |
|         // TODO: parsing format for DateTime etc
 | |
|         match fieldType with
 | |
|         | DateOnly ->
 | |
|             node
 | |
|             |> asValueGetValue propertyName "string"
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateLongIdent (SynLongIdent.Create [ "System" ; "DateOnly" ; "Parse" ])
 | |
|             )
 | |
|         | Uri ->
 | |
|             node
 | |
|             |> asValueGetValue propertyName "string"
 | |
|             |> SynExpr.pipeThroughFunction (SynExpr.CreateLongIdent (SynLongIdent.Create [ "System" ; "Uri" ]))
 | |
|         | DateTime ->
 | |
|             node
 | |
|             |> asValueGetValue propertyName "string"
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateLongIdent (SynLongIdent.Create [ "System" ; "DateTime" ; "Parse" ])
 | |
|             )
 | |
|         | NumberType typeName ->
 | |
|             let basic = asValueGetValue propertyName typeName node
 | |
| 
 | |
|             match options.JsonNumberHandlingArg with
 | |
|             | None -> basic
 | |
|             | Some option ->
 | |
|                 let cond =
 | |
|                     SynExpr.DotGet (
 | |
|                         SynExpr.CreateIdentString "exc",
 | |
|                         range0,
 | |
|                         SynLongIdent.CreateString "Message",
 | |
|                         range0
 | |
|                     )
 | |
|                     |> SynExpr.callMethodArg
 | |
|                         "Contains"
 | |
|                         (SynExpr.CreateConst (SynConst.CreateString "cannot be converted to"))
 | |
| 
 | |
|                 let handler =
 | |
|                     asValueGetValue propertyName "string" node
 | |
|                     |> SynExpr.pipeThroughFunction (
 | |
|                         SynExpr.CreateLongIdent (SynLongIdent.CreateFromLongIdent (parseFunction typeName))
 | |
|                     )
 | |
|                     |> SynExpr.ifThenElse
 | |
|                         (SynExpr.equals
 | |
|                             option
 | |
|                             (SynExpr.CreateLongIdent (
 | |
|                                 SynLongIdent.Create
 | |
|                                     [
 | |
|                                         "System"
 | |
|                                         "Text"
 | |
|                                         "Json"
 | |
|                                         "Serialization"
 | |
|                                         "JsonNumberHandling"
 | |
|                                         "AllowReadingFromString"
 | |
|                                     ]
 | |
|                             )))
 | |
|                         SynExpr.reraise
 | |
|                     |> SynExpr.ifThenElse cond SynExpr.reraise
 | |
| 
 | |
|                 basic
 | |
|                 |> SynExpr.pipeThroughTryWith
 | |
|                     (SynPat.IsInst (
 | |
|                         SynType.LongIdent (SynLongIdent.Create [ "System" ; "InvalidOperationException" ]),
 | |
|                         range0
 | |
|                     ))
 | |
|                     handler
 | |
|         | PrimitiveType typeName -> asValueGetValue propertyName typeName node
 | |
|         | OptionType ty ->
 | |
|             parseNode None options ty (SynExpr.CreateIdentString "v")
 | |
|             |> createParseLineOption node
 | |
|         | ListType ty ->
 | |
|             parseNode None options ty (SynExpr.CreateLongIdent (SynLongIdent.CreateString "elt"))
 | |
|             |> asArrayMapped propertyName "List" node
 | |
|         | ArrayType ty ->
 | |
|             parseNode None options ty (SynExpr.CreateLongIdent (SynLongIdent.CreateString "elt"))
 | |
|             |> asArrayMapped propertyName "Array" node
 | |
|         | IDictionaryType (keyType, valueType) ->
 | |
|             node
 | |
|             |> asObject propertyName
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateApp (
 | |
|                     SynExpr.CreateLongIdent (SynLongIdent.Create [ "Seq" ; "map" ]),
 | |
|                     dictionaryMapper (parseKeyString keyType) (parseNode None options valueType)
 | |
|                 )
 | |
|             )
 | |
|             |> SynExpr.pipeThroughFunction (SynExpr.CreateLongIdent (SynLongIdent.Create [ "dict" ]))
 | |
|         | DictionaryType (keyType, valueType) ->
 | |
|             node
 | |
|             |> asObject propertyName
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateApp (
 | |
|                     SynExpr.CreateLongIdent (SynLongIdent.Create [ "Seq" ; "map" ]),
 | |
|                     dictionaryMapper (parseKeyString keyType) (parseNode None options valueType)
 | |
|                 )
 | |
|             )
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateApp (
 | |
|                     SynExpr.CreateLongIdent (SynLongIdent.Create [ "Seq" ; "map" ]),
 | |
|                     SynExpr.CreateLongIdent (
 | |
|                         SynLongIdent.Create [ "System" ; "Collections" ; "Generic" ; "KeyValuePair" ]
 | |
|                     )
 | |
|                 )
 | |
|             )
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateLongIdent (SynLongIdent.Create [ "System" ; "Collections" ; "Generic" ; "Dictionary" ])
 | |
|             )
 | |
|         | IReadOnlyDictionaryType (keyType, valueType) ->
 | |
|             node
 | |
|             |> asObject propertyName
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateApp (
 | |
|                     SynExpr.CreateLongIdent (SynLongIdent.Create [ "Seq" ; "map" ]),
 | |
|                     dictionaryMapper (parseKeyString keyType) (parseNode None options valueType)
 | |
|                 )
 | |
|             )
 | |
|             |> SynExpr.pipeThroughFunction (SynExpr.CreateLongIdent (SynLongIdent.Create [ "readOnlyDict" ]))
 | |
|         | MapType (keyType, valueType) ->
 | |
|             node
 | |
|             |> asObject propertyName
 | |
|             |> SynExpr.pipeThroughFunction (
 | |
|                 SynExpr.CreateApp (
 | |
|                     SynExpr.CreateLongIdent (SynLongIdent.Create [ "Seq" ; "map" ]),
 | |
|                     dictionaryMapper (parseKeyString keyType) (parseNode None options valueType)
 | |
|                 )
 | |
|             )
 | |
|             |> SynExpr.pipeThroughFunction (SynExpr.CreateLongIdent (SynLongIdent.Create [ "Map" ; "ofSeq" ]))
 | |
|         | _ ->
 | |
|             // Let's just hope that we've also got our own type annotation!
 | |
|             let typeName =
 | |
|                 match fieldType with
 | |
|                 | SynType.LongIdent ident -> ident.LongIdent
 | |
|                 | _ -> failwith $"Unrecognised type: %+A{fieldType}"
 | |
| 
 | |
|             match propertyName with
 | |
|             | None -> node
 | |
|             | Some propertyName -> assertNotNull propertyName node
 | |
|             |> typeJsonParse typeName
 | |
| 
 | |
|     /// propertyName is probably a string literal, but it could be a [<Literal>] variable
 | |
|     /// The result of this function is the body of a let-binding (not including the LHS of that let-binding).
 | |
|     let createParseRhs (options : JsonParseOption) (propertyName : SynExpr) (fieldType : SynType) : SynExpr =
 | |
|         SynExpr.CreateIdentString "node"
 | |
|         |> SynExpr.index propertyName
 | |
|         |> parseNode (Some propertyName) options fieldType
 | |
| 
 | |
|     let isJsonNumberHandling (literal : LongIdent) : bool =
 | |
|         match List.rev literal |> List.map (fun ident -> ident.idText) with
 | |
|         | [ _ ; "JsonNumberHandling" ]
 | |
|         | [ _ ; "JsonNumberHandling" ; "Serialization" ]
 | |
|         | [ _ ; "JsonNumberHandling" ; "Serialization" ; "Json" ]
 | |
|         | [ _ ; "JsonNumberHandling" ; "Serialization" ; "Json" ; "Text" ]
 | |
|         | [ _ ; "JsonNumberHandling" ; "Serialization" ; "Json" ; "Text" ; "System" ] -> true
 | |
|         | _ -> false
 | |
| 
 | |
|     let createMaker (spec : JsonParseOutputSpec) (typeName : LongIdent) (fields : SynField list) =
 | |
|         let xmlDoc = PreXmlDoc.Create " Parse from a JSON node."
 | |
| 
 | |
|         let returnInfo =
 | |
|             SynBindingReturnInfo.Create (SynType.LongIdent (SynLongIdent.CreateFromLongIdent typeName))
 | |
| 
 | |
|         let inputArg = Ident.Create "node"
 | |
|         let functionName = Ident.Create "jsonParse"
 | |
| 
 | |
|         let inputVal =
 | |
|             let memberFlags =
 | |
|                 if spec.ExtensionMethods then
 | |
|                     {
 | |
|                         SynMemberFlags.IsInstance = false
 | |
|                         SynMemberFlags.IsDispatchSlot = false
 | |
|                         SynMemberFlags.IsOverrideOrExplicitImpl = false
 | |
|                         SynMemberFlags.IsFinal = false
 | |
|                         SynMemberFlags.GetterOrSetterIsCompilerGenerated = false
 | |
|                         SynMemberFlags.MemberKind = SynMemberKind.Member
 | |
|                     }
 | |
|                     |> Some
 | |
|                 else
 | |
|                     None
 | |
| 
 | |
|             let thisIdOpt = if spec.ExtensionMethods then None else Some inputArg
 | |
| 
 | |
|             SynValData.SynValData (
 | |
|                 memberFlags,
 | |
|                 SynValInfo.SynValInfo ([ [ SynArgInfo.CreateId functionName ] ], SynArgInfo.Empty),
 | |
|                 thisIdOpt
 | |
|             )
 | |
| 
 | |
|         let assignments =
 | |
|             fields
 | |
|             |> List.map (fun (SynField (attrs, _, id, fieldType, _, _, _, _, _)) ->
 | |
|                 let id =
 | |
|                     match id with
 | |
|                     | None -> failwith "didn't get an ID on field"
 | |
|                     | Some id -> id
 | |
| 
 | |
|                 let attrs = attrs |> List.collect (fun l -> l.Attributes)
 | |
| 
 | |
|                 let propertyNameAttr =
 | |
|                     attrs
 | |
|                     |> List.tryFind (fun attr ->
 | |
|                         attr.TypeName.AsString.EndsWith ("JsonPropertyName", StringComparison.Ordinal)
 | |
|                     )
 | |
| 
 | |
|                 let options =
 | |
|                     (JsonParseOption.None, attrs)
 | |
|                     ||> List.fold (fun options attr ->
 | |
|                         if attr.TypeName.AsString.EndsWith ("JsonNumberHandling", StringComparison.Ordinal) then
 | |
|                             let qualifiedEnumValue =
 | |
|                                 match SynExpr.stripOptionalParen attr.ArgExpr with
 | |
|                                 | SynExpr.LongIdent (_, SynLongIdent (ident, _, _), _, _) when
 | |
|                                     isJsonNumberHandling ident
 | |
|                                     ->
 | |
|                                     // Make sure it's fully qualified
 | |
|                                     SynExpr.CreateLongIdent (
 | |
|                                         SynLongIdent.Create
 | |
|                                             [
 | |
|                                                 "System"
 | |
|                                                 "Text"
 | |
|                                                 "Json"
 | |
|                                                 "Serialization"
 | |
|                                                 "JsonNumberHandling"
 | |
|                                                 "AllowReadingFromString"
 | |
|                                             ]
 | |
|                                     )
 | |
|                                 | _ -> attr.ArgExpr
 | |
| 
 | |
|                             {
 | |
|                                 JsonNumberHandlingArg = Some qualifiedEnumValue
 | |
|                             }
 | |
|                         else
 | |
|                             options
 | |
|                     )
 | |
| 
 | |
|                 let propertyName =
 | |
|                     match propertyNameAttr with
 | |
|                     | None ->
 | |
|                         let sb = StringBuilder id.idText.Length
 | |
|                         sb.Append (Char.ToLowerInvariant id.idText.[0]) |> ignore
 | |
| 
 | |
|                         if id.idText.Length > 1 then
 | |
|                             sb.Append id.idText.[1..] |> ignore
 | |
| 
 | |
|                         sb.ToString () |> SynConst.CreateString |> SynExpr.CreateConst
 | |
|                     | Some name -> name.ArgExpr
 | |
| 
 | |
|                 let pattern =
 | |
|                     SynPat.LongIdent (
 | |
|                         SynLongIdent.CreateFromLongIdent [ id ],
 | |
|                         None,
 | |
|                         None,
 | |
|                         SynArgPats.Empty,
 | |
|                         None,
 | |
|                         range0
 | |
|                     )
 | |
| 
 | |
|                 SynBinding.Let (
 | |
|                     isInline = false,
 | |
|                     isMutable = false,
 | |
|                     expr = createParseRhs options propertyName fieldType,
 | |
|                     valData = inputVal,
 | |
|                     pattern = pattern
 | |
|                 )
 | |
|             )
 | |
| 
 | |
|         let finalConstruction =
 | |
|             fields
 | |
|             |> List.map (fun (SynField (_, _, id, _, _, _, _, _, _)) ->
 | |
|                 let id =
 | |
|                     match id with
 | |
|                     | None -> failwith "Expected record field to have an identifying name"
 | |
|                     | Some id -> id
 | |
| 
 | |
|                 (SynLongIdent.CreateFromLongIdent [ id ], true),
 | |
|                 Some (SynExpr.CreateLongIdent (SynLongIdent.CreateFromLongIdent [ id ]))
 | |
|             )
 | |
|             |> AstHelper.instantiateRecord
 | |
| 
 | |
|         let assignments =
 | |
|             (finalConstruction, assignments)
 | |
|             ||> List.fold (fun final assignment ->
 | |
|                 SynExpr.LetOrUse (
 | |
|                     false,
 | |
|                     false,
 | |
|                     [ assignment ],
 | |
|                     final,
 | |
|                     range0,
 | |
|                     {
 | |
|                         InKeyword = None
 | |
|                     }
 | |
|                 )
 | |
|             )
 | |
| 
 | |
|         let pattern =
 | |
|             SynPat.LongIdent (
 | |
|                 SynLongIdent.CreateFromLongIdent [ functionName ],
 | |
|                 None,
 | |
|                 None,
 | |
|                 SynArgPats.Pats
 | |
|                     [
 | |
|                         SynPat.CreateTyped (
 | |
|                             SynPat.CreateNamed inputArg,
 | |
|                             SynType.LongIdent (
 | |
|                                 SynLongIdent.Create [ "System" ; "Text" ; "Json" ; "Nodes" ; "JsonNode" ]
 | |
|                             )
 | |
|                         )
 | |
|                         |> SynPat.CreateParen
 | |
|                     ],
 | |
|                 None,
 | |
|                 range0
 | |
|             )
 | |
| 
 | |
|         if spec.ExtensionMethods then
 | |
|             let binding =
 | |
|                 SynBinding.SynBinding (
 | |
|                     None,
 | |
|                     SynBindingKind.Normal,
 | |
|                     false,
 | |
|                     false,
 | |
|                     [],
 | |
|                     xmlDoc,
 | |
|                     inputVal,
 | |
|                     pattern,
 | |
|                     Some returnInfo,
 | |
|                     assignments,
 | |
|                     range0,
 | |
|                     DebugPointAtBinding.NoneAtInvisible,
 | |
|                     {
 | |
|                         LeadingKeyword = SynLeadingKeyword.StaticMember (range0, range0)
 | |
|                         InlineKeyword = None
 | |
|                         EqualsRange = Some range0
 | |
|                     }
 | |
|                 )
 | |
| 
 | |
|             let mem = SynMemberDefn.Member (binding, range0)
 | |
| 
 | |
|             let containingType =
 | |
|                 SynTypeDefn.SynTypeDefn (
 | |
|                     SynComponentInfo.Create (typeName, xmldoc = PreXmlDoc.Create " Extension methods for JSON parsing"),
 | |
|                     SynTypeDefnRepr.ObjectModel (SynTypeDefnKind.Augmentation range0, [], range0),
 | |
|                     [ mem ],
 | |
|                     None,
 | |
|                     range0,
 | |
|                     {
 | |
|                         LeadingKeyword = SynTypeDefnLeadingKeyword.Type range0
 | |
|                         EqualsRange = None
 | |
|                         WithKeyword = None
 | |
|                     }
 | |
|                 )
 | |
| 
 | |
|             SynModuleDecl.Types ([ containingType ], range0)
 | |
|         else
 | |
|             let binding =
 | |
|                 SynBinding.Let (
 | |
|                     isInline = false,
 | |
|                     isMutable = false,
 | |
|                     xmldoc = xmlDoc,
 | |
|                     returnInfo = returnInfo,
 | |
|                     expr = assignments,
 | |
|                     valData = inputVal,
 | |
|                     pattern = pattern
 | |
|                 )
 | |
| 
 | |
|             SynModuleDecl.CreateLet [ binding ]
 | |
| 
 | |
|     let createRecordModule (namespaceId : LongIdent) (spec : JsonParseOutputSpec) (typeDefn : SynTypeDefn) =
 | |
|         let (SynTypeDefn (synComponentInfo, synTypeDefnRepr, _members, _implicitCtor, _, _)) =
 | |
|             typeDefn
 | |
| 
 | |
|         let (SynComponentInfo (_attributes, _typeParams, _constraints, recordId, _, _preferPostfix, _access, _)) =
 | |
|             synComponentInfo
 | |
| 
 | |
|         match synTypeDefnRepr with
 | |
|         | SynTypeDefnRepr.Simple (SynTypeDefnSimpleRepr.Record (_accessibility, recordFields, _recordRange), _) ->
 | |
| 
 | |
|             let decls = [ createMaker spec recordId recordFields ]
 | |
| 
 | |
|             let attributes =
 | |
|                 if spec.ExtensionMethods then
 | |
|                     [ SynAttributeList.Create SynAttribute.autoOpen ]
 | |
|                 else
 | |
|                     [
 | |
|                         SynAttributeList.Create (SynAttribute.RequireQualifiedAccess ())
 | |
|                         SynAttributeList.Create SynAttribute.compilationRepresentation
 | |
|                     ]
 | |
| 
 | |
|             let xmlDoc =
 | |
|                 let fullyQualified = recordId |> Seq.map (fun i -> i.idText) |> String.concat "."
 | |
| 
 | |
|                 let description =
 | |
|                     if spec.ExtensionMethods then
 | |
|                         "extension members"
 | |
|                     else
 | |
|                         "methods"
 | |
| 
 | |
|                 $" Module containing JSON parsing %s{description} for the %s{fullyQualified} type"
 | |
|                 |> PreXmlDoc.Create
 | |
| 
 | |
|             let moduleName =
 | |
|                 if spec.ExtensionMethods then
 | |
|                     match recordId with
 | |
|                     | [] -> failwith "unexpectedly got an empty identifier for record name"
 | |
|                     | recordId ->
 | |
|                         let expanded =
 | |
|                             List.last recordId
 | |
|                             |> fun i -> i.idText
 | |
|                             |> fun s -> s + "JsonParseExtension"
 | |
|                             |> Ident.Create
 | |
| 
 | |
|                         List.take (List.length recordId - 1) recordId @ [ expanded ]
 | |
|                 else
 | |
|                     recordId
 | |
| 
 | |
|             let info =
 | |
|                 SynComponentInfo.Create (moduleName, attributes = attributes, xmldoc = xmlDoc)
 | |
| 
 | |
|             let mdl = SynModuleDecl.CreateNestedModule (info, decls)
 | |
| 
 | |
|             SynModuleOrNamespace.CreateNamespace (namespaceId, decls = [ mdl ])
 | |
|         | _ -> failwithf "Not a record type"
 | |
| 
 | |
| /// Myriad generator that provides a method (possibly an extension method) for a record type,
 | |
| /// containing a JSON parse function.
 | |
| [<MyriadGenerator("json-parse")>]
 | |
| type JsonParseGenerator () =
 | |
| 
 | |
|     interface IMyriadGenerator with
 | |
|         member _.ValidInputExtensions = [ ".fs" ]
 | |
| 
 | |
|         member _.Generate (context : GeneratorContext) =
 | |
|             let ast, _ =
 | |
|                 Ast.fromFilename context.InputFilename |> Async.RunSynchronously |> Array.head
 | |
| 
 | |
|             let records = Ast.extractRecords ast
 | |
| 
 | |
|             let namespaceAndRecords =
 | |
|                 records
 | |
|                 |> List.choose (fun (ns, types) ->
 | |
|                     types
 | |
|                     |> List.choose (fun typeDef ->
 | |
|                         match Ast.getAttribute<JsonParseAttribute> typeDef with
 | |
|                         | None -> None
 | |
|                         | Some attr ->
 | |
|                             let arg =
 | |
|                                 match SynExpr.stripOptionalParen attr.ArgExpr with
 | |
|                                 | SynExpr.Const (SynConst.Bool value, _) -> value
 | |
|                                 | SynExpr.Const (SynConst.Unit, _) -> JsonParseAttribute.DefaultIsExtensionMethod
 | |
|                                 | arg ->
 | |
|                                     failwith
 | |
|                                         $"Unrecognised argument %+A{arg} to [<%s{nameof JsonParseAttribute}>]. Literals are not supported. Use `true` or `false` (or unit) only."
 | |
| 
 | |
|                             let spec =
 | |
|                                 {
 | |
|                                     ExtensionMethods = arg
 | |
|                                 }
 | |
| 
 | |
|                             Some (typeDef, spec)
 | |
|                     )
 | |
|                     |> function
 | |
|                         | [] -> None
 | |
|                         | ty -> Some (ns, ty)
 | |
|                 )
 | |
| 
 | |
|             let modules =
 | |
|                 namespaceAndRecords
 | |
|                 |> List.collect (fun (ns, records) ->
 | |
|                     records
 | |
|                     |> List.map (fun (record, spec) ->
 | |
|                         let recordModule = JsonParseGenerator.createRecordModule ns spec record
 | |
|                         recordModule
 | |
|                     )
 | |
|                 )
 | |
| 
 | |
|             Output.Ast modules
 |