scribe/src/classes/page_converter.cr
Edward Loveall f05a12a880
Add support for missing posts
Posts, like 8661f4724aa9, can go missing if the account or post was
removed. In this case, the API returns data like this:

```json
{
  "data": {
    "post": null
  }
}
```

When this happens, we can detect it because the parsed response now has
a nil value: `response.data.post == nil` and construct an `EmptyPage`
instead of a `Page`. The `Articles::Show` action can then render
conditionally based on if the response from `PageConverter` is a `Page`
or an `EmptyPage`.
2022-06-17 16:00:01 -04:00

42 lines
1.3 KiB
Crystal

class PageConverter
def convert(post : PostResponse::Post) : Page
title, content = title_and_content(post)
author = post.creator
created_at = Time.unix_ms(post.createdAt)
gist_store = gist_store(content)
Page.new(
title: title,
author: author,
created_at: Time.unix_ms(post.createdAt),
nodes: ParagraphConverter.new.convert(content, gist_store)
)
end
def convert(post : Nil) : MissingPage
MissingPage.new
end
def title_and_content(post : PostResponse::Post) : {String, Array(PostResponse::Paragraph)}
title = post.title
paragraphs = post.content.bodyModel.paragraphs
non_content_paragraphs = paragraphs.reject { |para| para.text == title }
{title, non_content_paragraphs}
end
private def gist_store(paragraphs) : GistStore | RateLimitedGistStore
store = GistStore.new
gist_urls = GistScanner.new(paragraphs).scan
gist_responses = gist_urls.map do |url|
params = GistParams.extract_from_url(url)
response = GithubClient.get_gist_response(params.id)
if response.is_a?(GithubClient::RateLimitedResponse)
return RateLimitedGistStore.new
end
JSON.parse(response.data.body)["files"].as_h.values.map do |json_any|
store.store_gist_file(params.id, GistFile.from_json(json_any.to_json))
end
end
store
end
end