hydraNormalizer.ts 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. import {AnyJson, ApiResponse, HydraMetadata} from '~/types/data'
  2. import UrlUtils from '~/services/utils/urlUtils'
  3. import {METADATA_TYPE} from '~/types/enum/data'
  4. import models from "~/models/models";
  5. import ApiResource from "~/models/ApiResource";
  6. import {isArray} from "lodash";
  7. /**
  8. * Normalisation et dé-normalisation du format de données Hydra
  9. */
  10. class HydraNormalizer {
  11. /**
  12. * Normalize the given entity into an Hydra formatted content.
  13. * @param entity
  14. */
  15. public static normalizeEntity(entity: ApiResource): AnyJson {
  16. const iriEncodedFields = HydraNormalizer.getIriEncodedFields(entity)
  17. for (const field in iriEncodedFields) {
  18. const value = entity[field]
  19. const targetEntity = iriEncodedFields[field].entity
  20. if (isArray(value)) {
  21. entity[field] = value.map((id: number) => {
  22. return UrlUtils.makeIRI(targetEntity, id)
  23. })
  24. } else {
  25. entity[field] = UrlUtils.makeIRI(targetEntity, value)
  26. }
  27. }
  28. return entity.$toJson()
  29. }
  30. /**
  31. * Parse une réponse Hydra et retourne un objet ApiResponse
  32. *
  33. * @param {AnyJson} data
  34. * @param model
  35. * @return {AnyJson} réponse parsée
  36. */
  37. public static denormalize(data: AnyJson, model?: typeof ApiResource): ApiResponse {
  38. return {
  39. data: HydraNormalizer.getData(data, model),
  40. metadata: HydraNormalizer.getMetadata(data)
  41. }
  42. }
  43. protected static getData(hydraData: AnyJson, model?: typeof ApiResource): AnyJson {
  44. if (hydraData['@type'] === 'hydra:Collection') {
  45. let members = hydraData['hydra:member']
  46. let results = []
  47. for (let item of members) {
  48. results.push(HydraNormalizer.denormalizeItem(item, model))
  49. }
  50. return results
  51. } else {
  52. return HydraNormalizer.denormalizeItem(hydraData, model)
  53. }
  54. }
  55. /**
  56. * Dénormalise un item d'une réponse hydra
  57. *
  58. * @param item
  59. * @param model
  60. * @protected
  61. */
  62. protected static denormalizeItem(item: AnyJson, model?: typeof ApiResource): AnyJson {
  63. if (!model) {
  64. const {entity, id} = HydraNormalizer.parseIRI(item['@id'])
  65. if (!id) {
  66. throw Error('De-normalization error : Could not determine the model of the entity')
  67. }
  68. model = models[entity]
  69. }
  70. const instance = new model(item)
  71. const iriEncodedFields = HydraNormalizer.getIriEncodedFields(instance)
  72. for (const field in iriEncodedFields) {
  73. const value = instance[field]
  74. const targetEntity = iriEncodedFields[field].entity
  75. if (isArray(value)) {
  76. instance[field] = value.map((iri: string) => {
  77. return HydraNormalizer.getIdFromIri(iri, targetEntity)
  78. })
  79. } else {
  80. instance[field] = HydraNormalizer.getIdFromIri(value, targetEntity)
  81. }
  82. }
  83. return instance
  84. }
  85. /**
  86. * Génère les métadonnées d'un item ou d'une collection
  87. *
  88. * @param data
  89. * @protected
  90. */
  91. protected static getMetadata(data: AnyJson): HydraMetadata {
  92. if (data['@type'] !== 'hydra:Collection') {
  93. // A single item, no metadata
  94. return { type: METADATA_TYPE.ITEM }
  95. }
  96. const metadata: HydraMetadata = {
  97. totalItems: data['hydra:totalItems']
  98. }
  99. if (data['hydra:view']) {
  100. /**
  101. * Extract the page number from the IRIs in the hydra:view section
  102. */
  103. const extractPageNumber = (pos: string, default_: number | undefined=undefined): number | undefined => {
  104. const iri = data['hydra:view']['hydra:' + pos]
  105. if (!iri) {
  106. return default_
  107. }
  108. return UrlUtils.getParameter(
  109. data['hydra:view']['hydra:' + pos],
  110. 'page',
  111. default_
  112. ) as number | undefined
  113. }
  114. // TODO: utile d'ajouter la page en cours?
  115. metadata.firstPage = extractPageNumber('first', 1)
  116. metadata.lastPage = extractPageNumber('last', 1)
  117. metadata.nextPage = extractPageNumber('next')
  118. metadata.previousPage = extractPageNumber('previous')
  119. }
  120. metadata.type = METADATA_TYPE.COLLECTION
  121. return metadata
  122. }
  123. /**
  124. * Parse the given IRI
  125. * @param iri
  126. * @protected
  127. */
  128. protected static parseIRI(iri: string) {
  129. const rx = /\/api\/(\w+)\/(\d+)/
  130. const match = rx.exec(iri)
  131. if (!match) {
  132. throw Error('could not parse the IRI : ' + iri)
  133. }
  134. return {
  135. entity: match[1],
  136. id: parseInt(match[2])
  137. }
  138. }
  139. /**
  140. * Get the array of the entity's fields marked as IRIEncoded
  141. * @see models/decorators.ts
  142. *
  143. * @param entity
  144. * @protected
  145. */
  146. protected static getIriEncodedFields(entity: ApiResource): Record<string, ApiResource> {
  147. const prototype = Object.getPrototypeOf(entity)
  148. return prototype.constructor.relations
  149. }
  150. /**
  151. * Retrieve the entitie's id from the given IRI
  152. * Throws an error if the IRI does not match the expected entity
  153. *
  154. * @param iri
  155. * @param expectedEntity
  156. * @protected
  157. */
  158. protected static getIdFromIri(iri: string, expectedEntity: string): number {
  159. const { entity, id } = HydraNormalizer.parseIRI(iri)
  160. if (entity !== expectedEntity) {
  161. throw Error("IRI entity does not match the field's target entity (" + entity + ' != ' + expectedEntity + ")")
  162. }
  163. return id
  164. }
  165. }
  166. export default HydraNormalizer