Skip to content

Commit

Permalink
Replace links in "fields" with linked to entites from "includes"
Browse files Browse the repository at this point in the history
- Resolve all link fields found within "items" in an API response, and replace the links with the actual entities if available from the "includes" section of API response. Inspired by https://github.com/contentful/contentful.js/blob/master/ADVANCED.md#link-resolution
- Breaking change: 'assets' field on Entry struct is no longer populated. All Assets for an Entry are embedded directly within 'fields' now instead
  • Loading branch information
WillRochaThomas committed May 6, 2024
1 parent c90493c commit a24a142
Show file tree
Hide file tree
Showing 11 changed files with 726 additions and 256 deletions.
65 changes: 35 additions & 30 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,83 +2,88 @@

## Unreleased

## 1.0.0

- Resolve all link fields found within "items" in an API response, and replace the links with the actual entities if available from the "includes" section of API response. Inspired by https://github.com/contentful/contentful.js/blob/master/ADVANCED.md#link-resolution
- Breaking change: 'assets' field on Entry struct is no longer populated. All Assets for an Entry are embedded directly within 'fields' now instead

## 0.5.0

* [#97](https://github.com/contentful-userland/contentful.ex/pull/97) switches underlying http adapter to `tesla` for higher flexibility. Thank you @OldhamMade
* tests against Elixir 1.13 and updates required elixir version to 1.11
- [#97](https://github.com/contentful-userland/contentful.ex/pull/97) switches underlying http adapter to `tesla` for higher flexibility. Thank you @OldhamMade
- tests against Elixir 1.13 and updates required elixir version to 1.11

## 0.4.1

* [#75](https://github.com/contentful-labs/contentful.ex/issue/75) fixes an issue concerning malformed docs, thanks @OldhamMade
- [#75](https://github.com/contentful-labs/contentful.ex/issue/75) fixes an issue concerning malformed docs, thanks @OldhamMade

## 0.4.0

* [#49](https://github.com/contentful-labs/contentful.ex/pull/49) Adds extended query syntax for building more complex queries, as suggested by @ryansch in [#38](https://github.com/contentful-labs/contentful.ex/issues/38)
* adds testing against Elixir 1.10.4
- [#49](https://github.com/contentful-labs/contentful.ex/pull/49) Adds extended query syntax for building more complex queries, as suggested by @ryansch in [#38](https://github.com/contentful-labs/contentful.ex/issues/38)
- adds testing against Elixir 1.10.4

## 0.3.2

* [#47](https://github.com/contentful-labs/contentful.ex/pull/47) Handle bitstring case when resolving assets (thanks @aspala)
- [#47](https://github.com/contentful-labs/contentful.ex/pull/47) Handle bitstring case when resolving assets (thanks @aspala)

## 0.3.1

* [#37](https://github.com/contentful-labs/contentful.ex/issues/37) Fixed an error preventing correct entity resolution for assets (thanks @OldhamMade)
* [#44](https://github.com/contentful-labs/contentful.ex/issues/44) Adds missing common properties to content types, assets entries (thanks @OldhamMade)
* [#36](https://github.com/contentful-labs/contentful.ex/issues/36) Added dependabot for keeping dependencies up to date
* [#9](https://github.com/contentful-labs/contentful.ex/issues/9) Added the ability to specify an endpoint other than the Delivery API
* Improved some README sections about how to query certain entities
- [#37](https://github.com/contentful-labs/contentful.ex/issues/37) Fixed an error preventing correct entity resolution for assets (thanks @OldhamMade)
- [#44](https://github.com/contentful-labs/contentful.ex/issues/44) Adds missing common properties to content types, assets entries (thanks @OldhamMade)
- [#36](https://github.com/contentful-labs/contentful.ex/issues/36) Added dependabot for keeping dependencies up to date
- [#9](https://github.com/contentful-labs/contentful.ex/issues/9) Added the ability to specify an endpoint other than the Delivery API
- Improved some README sections about how to query certain entities

## 0.3.0

### Features

* Introduced a DSL that can be composed into queries
* Solved the `include` removal by readding it (#20)
* Also allows for resolving assets included in entries
- Introduced a DSL that can be composed into queries
- Solved the `include` removal by readding it (#20)
- Also allows for resolving assets included in entries

### Chores

* Deleted most Context(s) module code
* Updated the docs with more working examples
- Deleted most Context(s) module code
- Updated the docs with more working examples

## 0.2.0

Note: This release is incompatible with previous releases as this lacks the `include` functionality, specifically.

### Features

* Reworked the way data can be queried from the CDA endpoint
* Split up the modules into mapping to different Contentful APIs
* Introduced a way to stream the CDA API endpoints instead of relying on pagination
* Added
- Reworked the way data can be queried from the CDA endpoint
- Split up the modules into mapping to different Contentful APIs
- Introduced a way to stream the CDA API endpoints instead of relying on pagination
- Added

### Chores

* Added badges and `ex_doc` integration (published via [hex.pm](https://hex.pm))
* Updated the docs with examples
- Added badges and `ex_doc` integration (published via [hex.pm](https://hex.pm))
- Updated the docs with examples

## 0.1.1

### Fixed

* Fixed issue on empty includes
* Fixed compatibility with Elixir 1.4
- Fixed issue on empty includes
- Fixed compatibility with Elixir 1.4

## 0.1.0

### Added

* Added some spec coverage for all endpoints and include resolution
* Added Linter tool
- Added some spec coverage for all endpoints and include resolution
- Added Linter tool

### Changed

* Improved syntax conventions
* Refactored Include Resolution into it's own module
- Improved syntax conventions
- Refactored Include Resolution into it's own module

## 0.0.1 [INITIAL RELEASE]

### Added

* Added all CDA Endpoints
* Added Basic Include Resolution
- Added all CDA Endpoints
- Added Basic Include Resolution
57 changes: 57 additions & 0 deletions fixture/vcr_cassettes/some_entries_have_links.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
[
{
"request": {
"body": "",
"headers": {
"authorization": "***",
"User-Agent": "Contentful Elixir SDK",
"X-Contentful-User-Agent": "Contentful Elixir SDK",
"accept": "application/json"
},
"method": "get",
"options": {
"httpc_options": [],
"http_options": {
"autoredirect": "false"
}
},
"request_body": "",
"url": "https://cdn.contentful.com/spaces/bmehzfuz4raf/environments/master/entries?content_type=blogPost&sys.id=2PtC9h1YqIA6kaUaIsWEQ0"
},
"response": {
"binary": false,
"body": "{\"sys\":{\"type\":\"Array\"},\"total\":1,\"skip\":0,\"limit\":100,\"items\":[{\"metadata\":{\"tags\":[]},\"sys\":{\"space\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"Space\",\"id\":\"bmehzfuz4raf\"}},\"id\":\"2PtC9h1YqIA6kaUaIsWEQ0\",\"type\":\"Entry\",\"createdAt\":\"2019-03-22T08:33:45.069Z\",\"updatedAt\":\"2020-04-18T18:44:10.843Z\",\"environment\":{\"sys\":{\"id\":\"master\",\"type\":\"Link\",\"linkType\":\"Environment\"}},\"revision\":2,\"contentType\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"ContentType\",\"id\":\"blogPost\"}},\"locale\":\"en-US\"},\"fields\":{\"title\":\"Static sites are great\",\"slug\":\"static-sites-are-great\",\"heroImage\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"Asset\",\"id\":\"4NzwDSDlGECGIiokKomsyI\"}},\"description\":\"Worry less about security, caching, and talking to the server. Static sites are the new thing.\",\"body\":\"## The case for the static site generator\\n\\nMore and more developers are jumping on the \\\"go static train\\\", and rightfully so. Static pages are fast, lightweight, they scale well. They are more secure, and simple to maintain and they allow you to focus all your time and effort on the user interface. Often times, this dedication really shows.\\n\\nIt just so happens that static site generators are mostly loved by developers, but not by the average Joe. They do not offer WYSIWYG, previewing on demo sites may take an update cycle, they are often based on markdown text files, and they require some knowledge of modern day repositories.\\n\\nMoreover, when teams are collaborating, it can get complicated quickly. Has this article already been proof-read or reviewed? Is this input valid? Are user permissions available, e.g. for administering adding and removing team members? Can this article be published at a future date? How can a large repository of content be categorized, organized, and searched? All these requirements have previously been more or less solved within the admin area of your CMS. But of course with all the baggage that made you leave the appserver-app-database-in-one-big-blob stack in the first place.\\n\\n## Content APIs to the rescue\\n\\nAn alternative is decoupling the content management aspect from the system. And then replacing the maintenance prone server with a cloud based web service offering. Effectively, instead of your CMS of old, you move to a [Content Management as a Service (CMaaS)](https://www.contentful.com/r/knowledgebase/content-as-a-service/ \\\"Content Management as a Service (CMaaS)\\\") world, with a content API to deliver all your content. That way, you get the all the [benefits of content management features](http://www.digett.com/blog/01/16/2014/pairing-static-websites-cms \\\"benefits of content management features\\\") while still being able to embrace the static site generator mantra.\\n\\nIt so happens that Contentful is offering just that kind of content API. A service that\\n\\n* from the ground up has been designed to be fast, scalable, secure, and offer high uptime, so that you don’t have to worry about maintenance ever again.\\n* offers a powerful editor and lots of flexibility in creating templates for your documents that your editors can reuse and combine, so that no developers resources are required in everyday writing and updating tasks.\\n* separates content from presentation, so you can reuse your content repository for any device platform your heart desires. That way, you can COPE (\\\"create once, publish everywhere\\\").\\n* offers webhooks that you can use to rebuild your static site in a fully automated fashion every time your content is modified.\\n\\nExtracted from the article [CMS-functionality for static site generators](https://www.contentful.com/r/knowledgebase/contentful-api-cms-static-site-generators/ \\\"CMS-functionality for static site generators\\\"). Read more about the [static site generators supported by Contentful](https://www.contentful.com/developers/docs/tools/staticsitegenerators/ \\\"static site generators supported by Contentful\\\").\",\"author\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"Entry\",\"id\":\"15jwOBqpxqSAOy2eOO4S0m\"}},\"publishDate\":\"2017-05-16T00:00+02:00\",\"tags\":[\"javascript\",\"static-sites\"]}}],\"includes\":{\"Entry\":[{\"metadata\":{\"tags\":[]},\"sys\":{\"space\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"Space\",\"id\":\"bmehzfuz4raf\"}},\"id\":\"15jwOBqpxqSAOy2eOO4S0m\",\"type\":\"Entry\",\"createdAt\":\"2019-03-22T08:33:44.329Z\",\"updatedAt\":\"2020-04-18T18:44:10.435Z\",\"environment\":{\"sys\":{\"id\":\"master\",\"type\":\"Link\",\"linkType\":\"Environment\"}},\"revision\":2,\"contentType\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"ContentType\",\"id\":\"person\"}},\"locale\":\"en-US\"},\"fields\":{\"name\":\"John Doe\",\"title\":\"Web Developer\",\"company\":\"ACME\",\"shortBio\":\"Research and recommendations for modern stack websites.\",\"email\":\"[email protected]\",\"phone\":\"0176 / 1234567\",\"facebook\":\"johndoe\",\"twitter\":\"johndoe\",\"github\":\"johndoe\",\"image\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"Asset\",\"id\":\"7orLdboQQowIUs22KAW4U\"}}}}],\"Asset\":[{\"metadata\":{\"tags\":[]},\"sys\":{\"space\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"Space\",\"id\":\"bmehzfuz4raf\"}},\"id\":\"4NzwDSDlGECGIiokKomsyI\",\"type\":\"Asset\",\"createdAt\":\"2019-03-22T08:33:39.477Z\",\"updatedAt\":\"2020-04-18T18:44:05.885Z\",\"environment\":{\"sys\":{\"id\":\"master\",\"type\":\"Link\",\"linkType\":\"Environment\"}},\"revision\":2,\"locale\":\"en-US\"},\"fields\":{\"title\":\"City\",\"description\":\"City pictured from the sky\",\"file\":{\"url\":\"//images.ctfassets.net/bmehzfuz4raf/4NzwDSDlGECGIiokKomsyI/90b101c671a5969764757aaefb0c3466/denys-nevozhai-100695.jpg\",\"details\":{\"size\":15736986,\"image\":{\"width\":3992,\"height\":2992}},\"fileName\":\"denys-nevozhai-100695.jpg\",\"contentType\":\"image/jpeg\"}}},{\"metadata\":{\"tags\":[]},\"sys\":{\"space\":{\"sys\":{\"type\":\"Link\",\"linkType\":\"Space\",\"id\":\"bmehzfuz4raf\"}},\"id\":\"7orLdboQQowIUs22KAW4U\",\"type\":\"Asset\",\"createdAt\":\"2019-03-22T08:33:38.110Z\",\"updatedAt\":\"2020-04-18T18:44:04.820Z\",\"environment\":{\"sys\":{\"id\":\"master\",\"type\":\"Link\",\"linkType\":\"Environment\"}},\"revision\":2,\"locale\":\"en-US\"},\"fields\":{\"title\":\"Sparkler\",\"description\":\"John with Sparkler\",\"file\":{\"url\":\"//images.ctfassets.net/bmehzfuz4raf/7orLdboQQowIUs22KAW4U/ae1e04accdfcf6c3def7a449d12bff4c/matt-palmer-254999.jpg\",\"details\":{\"size\":2293094,\"image\":{\"width\":3000,\"height\":2000}},\"fileName\":\"matt-palmer-254999.jpg\",\"contentType\":\"image/jpeg\"}}}]}}\n",
"headers": {
"connection": "keep-alive",
"date": "Tue, 30 Apr 2024 08:46:13 GMT",
"via": "1.1 varnish, 1.1 varnish",
"accept-ranges": "bytes",
"age": "207",
"etag": "\"6650198899795356495\"",
"server": "Contentful",
"content-length": "6149",
"content-type": "application/vnd.contentful.delivery.v1+json",
"cf-space-id": "bmehzfuz4raf",
"cf-environment-id": "master",
"cf-environment-uuid": "fe6aa454-82d9-4c41-83f9-5f7194ff7797",
"cf-organization-id": "42rCduzn2g6MhnFOjK2VMh",
"x-contentful-route": "/spaces/:space/environments/:environment/entries",
"x-content-type-options": "nosniff",
"contentful-api": "cda",
"x-contentful-region": "us-east-1",
"access-control-allow-origin": "*",
"access-control-allow-headers": "Accept,Accept-Language,Authorization,Cache-Control,Content-Length,Content-Range,Content-Type,DNT,Destination,Expires,If-Match,If-Modified-Since,If-None-Match,Keep-Alive,Last-Modified,Origin,Pragma,Range,User-Agent,X-Http-Method-Override,X-Mx-ReqToken,X-Requested-With,X-Contentful-Version,X-Contentful-Content-Type,X-Contentful-Organization,X-Contentful-Skip-Transformation,X-Contentful-User-Agent,X-Contentful-Enable-Alpha-Feature,X-Contentful-Resource-Resolution",
"access-control-expose-headers": "Etag",
"access-control-max-age": "86400",
"access-control-allow-methods": "GET,HEAD,OPTIONS",
"x-served-by": "cache-ewr18121-EWR, cache-lcy-eglc8600094-LCY",
"x-cache-hits": "0, 0",
"x-timer": "S1714466774.847754,VS0,VE1",
"x-cache": "HIT",
"x-contentful-request-id": "c0a1e440-26e0-468c-a225-cfb8c6c5230a"
},
"status_code": ["HTTP/1.1", 200, "OK"],
"type": "ok"
}
}
]
12 changes: 8 additions & 4 deletions lib/contentful/entry.ex
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,17 @@ defmodule Contentful.Entry do
See the [official documentation for more information](https://www.contentful.com/developers/docs/references/content-delivery-api/#/reference/entries).
"""

alias Contentful.{Asset, SysData}
alias Contentful.SysData

defstruct [:sys, fields: [], assets: []]
defstruct [:sys, fields: []]

@type t :: %Contentful.Entry{
fields: list(),
sys: SysData.t(),
assets: list(Asset.t())
sys: SysData.t()
}

@deprecated "assets are no longer populated on an Entry, instead all links within 'fields' are replaced automatically with actual entities (if available in 'includes')"
def assets() do
[]
end
end
54 changes: 0 additions & 54 deletions lib/contentful/entry/asset_resolver.ex

This file was deleted.

Loading

0 comments on commit a24a142

Please sign in to comment.