feat(users/Profpatsch/whatcd-resolver): render anon json+ld objs
Some json+ld objects cannot be expanded any further, they are missing the `@id` tag. Now we also render them as objects. Change-Id: I1c8f26f3c34e69420c349e66a3ce5a36dc55a1ea Reviewed-on: https://cl.tvl.fyi/c/depot/+/11173 Reviewed-by: Profpatsch <mail@profpatsch.de> Autosubmit: Profpatsch <mail@profpatsch.de> Tested-by: BuildkiteCI
This commit is contained in:
parent
3b9fb1aa60
commit
32c5674962
1 changed files with 66 additions and 42 deletions
|
@ -453,6 +453,7 @@ getBestTorrentsTable = do
|
||||||
-- | A recursive `json+ld` structure.
|
-- | A recursive `json+ld` structure.
|
||||||
data Jsonld
|
data Jsonld
|
||||||
= JsonldObject JsonldObject
|
= JsonldObject JsonldObject
|
||||||
|
| JsonldAnonymousObject JsonldAnonymousObject
|
||||||
| JsonldArray [Jsonld]
|
| JsonldArray [Jsonld]
|
||||||
| JsonldField Json.Value
|
| JsonldField Json.Value
|
||||||
deriving stock (Show, Eq)
|
deriving stock (Show, Eq)
|
||||||
|
@ -468,64 +469,87 @@ data JsonldObject = JsonldObject'
|
||||||
}
|
}
|
||||||
deriving stock (Show, Eq)
|
deriving stock (Show, Eq)
|
||||||
|
|
||||||
|
-- | A json+ld object that cannot be inspected further by resolving its ID
|
||||||
|
data JsonldAnonymousObject = JsonldAnonymousObject'
|
||||||
|
{ -- | `@type` field; currently just the plain value without taking into account the json+ld context
|
||||||
|
type_ :: Set Text,
|
||||||
|
-- | fields of this anonymous object
|
||||||
|
fields :: Map Text Jsonld
|
||||||
|
}
|
||||||
|
deriving stock (Show, Eq)
|
||||||
|
|
||||||
jsonldParser :: (Monad m) => Json.ParseT err m Jsonld
|
jsonldParser :: (Monad m) => Json.ParseT err m Jsonld
|
||||||
jsonldParser =
|
jsonldParser =
|
||||||
Json.asValue >>= \cur -> do
|
Json.asValue >>= \cur -> do
|
||||||
if
|
if
|
||||||
| Json.Object _ <- cur -> do
|
| Json.Object _ <- cur -> do
|
||||||
typeMay <- Json.keyMay "@type" $ (Json.asArraySet Json.asText Json.<|> (Set.singleton <$> Json.asText))
|
type_ <-
|
||||||
|
Json.keyMay "@type" (Json.asArraySet Json.asText Json.<|> (Set.singleton <$> Json.asText))
|
||||||
|
<&> fromMaybe Set.empty
|
||||||
idMay <- Json.keyMay "@id" $ Json.asText
|
idMay <- Json.keyMay "@id" $ Json.asText
|
||||||
|
fields <-
|
||||||
|
Json.asObjectMap jsonldParser
|
||||||
|
<&> Map.delete "@type"
|
||||||
|
<&> Map.delete "@id"
|
||||||
|
|
||||||
if
|
if
|
||||||
| Just type_ <- typeMay,
|
| Just id_ <- idMay -> do
|
||||||
Just id_ <- idMay -> do
|
pure $ JsonldObject $ JsonldObject' {previewFields = fields, ..}
|
||||||
previewFields <-
|
| otherwise -> pure $ JsonldAnonymousObject $ JsonldAnonymousObject' {..}
|
||||||
Json.asObjectMap jsonldParser
|
|
||||||
<&> Map.delete "@type"
|
|
||||||
<&> Map.delete "@id"
|
|
||||||
pure $ JsonldObject $ JsonldObject' {..}
|
|
||||||
| otherwise -> pure $ JsonldField cur
|
|
||||||
| Json.Array _ <- cur -> do
|
| Json.Array _ <- cur -> do
|
||||||
JsonldArray <$> Json.eachInArray jsonldParser
|
JsonldArray <$> Json.eachInArray jsonldParser
|
||||||
| otherwise -> pure $ JsonldField cur
|
| otherwise -> pure $ JsonldField cur
|
||||||
|
|
||||||
renderJsonld :: Jsonld -> Html
|
renderJsonld :: Jsonld -> Html
|
||||||
renderJsonld = \case
|
renderJsonld = \case
|
||||||
JsonldObject obj ->
|
JsonldObject obj -> renderObject obj (Just obj.id_) obj.previewFields
|
||||||
[hsx|
|
JsonldAnonymousObject obj -> renderObject obj Nothing obj.fields
|
||||||
<dl>
|
|
||||||
<dt>Type</dt>
|
|
||||||
<dd>{obj.type_ & toList & schemaTypes}</dd>
|
|
||||||
<dt>Url</dt>
|
|
||||||
<dd><a href={obj.id_}>{obj.id_}</a></dd>
|
|
||||||
<dt>Fields</dt>
|
|
||||||
<dd>
|
|
||||||
{obj.previewFields & Html.toDefinitionList schemaType renderJsonld}
|
|
||||||
<div>
|
|
||||||
<button
|
|
||||||
hx-get={snippetHref obj.id_}
|
|
||||||
hx-target="closest dl"
|
|
||||||
hx-swap="outerHTML"
|
|
||||||
>more fields …</button>
|
|
||||||
</div>
|
|
||||||
</dd>
|
|
||||||
</dl>
|
|
||||||
|]
|
|
||||||
where
|
|
||||||
snippetHref target =
|
|
||||||
Builder.toLazyByteString $
|
|
||||||
"/snips/jsonld/render"
|
|
||||||
<> Url.renderQueryBuilder True [("target", Just (textToBytesUtf8 target))]
|
|
||||||
|
|
||||||
schemaTypes xs =
|
|
||||||
xs
|
|
||||||
<&> schemaType
|
|
||||||
& List.intersperse ", "
|
|
||||||
& mconcat
|
|
||||||
schemaType t =
|
|
||||||
let href :: Text = [fmt|https://schema.org/{t}|] in [hsx|<a href={href} target="_blank">{t}</a>|]
|
|
||||||
JsonldArray arr ->
|
JsonldArray arr ->
|
||||||
Html.toOrderedList renderJsonld arr
|
Html.toOrderedList renderJsonld arr
|
||||||
JsonldField f -> Html.mkVal f
|
JsonldField f -> Html.mkVal f
|
||||||
|
where
|
||||||
|
renderObject obj mId_ fields = do
|
||||||
|
let id_ =
|
||||||
|
mId_ <&> \i ->
|
||||||
|
[hsx|
|
||||||
|
<dt>Url</dt>
|
||||||
|
<dd><a href={i}>{i}</a></dd>
|
||||||
|
|]
|
||||||
|
getMoreButton =
|
||||||
|
mId_ <&> \i ->
|
||||||
|
[hsx|
|
||||||
|
<div>
|
||||||
|
<button
|
||||||
|
hx-get={snippetHref i}
|
||||||
|
hx-target="closest dl"
|
||||||
|
hx-swap="outerHTML"
|
||||||
|
>more fields …</button>
|
||||||
|
</div>
|
||||||
|
|]
|
||||||
|
[hsx|
|
||||||
|
<dl>
|
||||||
|
<dt>Type</dt>
|
||||||
|
<dd>{obj.type_ & toList & schemaTypes}</dd>
|
||||||
|
{id_}
|
||||||
|
<dt>Fields</dt>
|
||||||
|
<dd>
|
||||||
|
{fields & Html.toDefinitionList schemaType renderJsonld}
|
||||||
|
{getMoreButton}
|
||||||
|
</dd>
|
||||||
|
</dl>
|
||||||
|
|]
|
||||||
|
snippetHref target =
|
||||||
|
Builder.toLazyByteString $
|
||||||
|
"/snips/jsonld/render"
|
||||||
|
<> Url.renderQueryBuilder True [("target", Just (textToBytesUtf8 target))]
|
||||||
|
|
||||||
|
schemaTypes xs =
|
||||||
|
xs
|
||||||
|
<&> schemaType
|
||||||
|
& List.intersperse ", "
|
||||||
|
& mconcat
|
||||||
|
schemaType t =
|
||||||
|
let href :: Text = [fmt|https://schema.org/{t}|] in [hsx|<a href={href} target="_blank">{t}</a>|]
|
||||||
|
|
||||||
-- | A value between (inclusive) 0% and (inclusive) 100%. Precise to 1% steps.
|
-- | A value between (inclusive) 0% and (inclusive) 100%. Precise to 1% steps.
|
||||||
newtype Percentage = Percentage {unPercentage :: Int}
|
newtype Percentage = Percentage {unPercentage :: Int}
|
||||||
|
|
Loading…
Reference in a new issue