utils.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. // TODO: finish db cascade on save/remove; right now there is manually added `cascade: ["insert", "update"]` directive
  2. // to all relations in `query-node/generated/graphql-server/src/modules/**/*.model.ts`. That should ensure all records
  3. // are saved on one `store.save(...)` call. Missing features
  4. // - find a proper way to cascade on remove or implement custom removals for every entity
  5. // - convert manual changes done to `*model.ts` file into some patch or bash commands that can be executed
  6. // every time query node codegen is run (that will overwrite said manual changes)
  7. // - verify in integration tests that the records are trully created/updated/removed as expected
  8. import { DatabaseManager, EventContext, StoreContext } from '@joystream/hydra-common'
  9. import ISO6391 from 'iso-639-1'
  10. import { FindConditions } from 'typeorm'
  11. import {
  12. IVideoMetadata,
  13. IPublishedBeforeJoystream,
  14. ILicense,
  15. IMediaType,
  16. IChannelMetadata,
  17. } from '@joystream/metadata-protobuf'
  18. import { integrateMeta, isSet } from '@joystream/metadata-protobuf/utils'
  19. import { invalidMetadata, inconsistentState, logger, unexpectedData, createDataObject } from '../common'
  20. import {
  21. // primary entities
  22. CuratorGroup,
  23. Channel,
  24. Video,
  25. VideoCategory,
  26. // secondary entities
  27. Language,
  28. License,
  29. VideoMediaMetadata,
  30. // asset
  31. Asset,
  32. DataObjectOwner,
  33. DataObjectOwnerChannel,
  34. Membership,
  35. VideoMediaEncoding,
  36. AssetExternal,
  37. AssetJoystreamStorage,
  38. ChannelCategory,
  39. } from 'query-node/dist/model'
  40. // Joystream types
  41. import { ContentParameters, NewAsset, ContentActor } from '@joystream/types/augment'
  42. import { ContentParameters as Custom_ContentParameters } from '@joystream/types/storage'
  43. import { registry } from '@joystream/types'
  44. import { DecodedMetadataObject } from '@joystream/metadata-protobuf/types'
  45. import BN from 'bn.js'
  46. export async function processChannelMetadata(
  47. ctx: EventContext & StoreContext,
  48. channel: Channel,
  49. meta: DecodedMetadataObject<IChannelMetadata>,
  50. assets: NewAsset[]
  51. ): Promise<Channel> {
  52. const assetsOwner = new DataObjectOwnerChannel()
  53. assetsOwner.channelId = channel.id
  54. const processedAssets = await Promise.all(assets.map((asset) => processNewAsset(ctx, asset, assetsOwner)))
  55. integrateMeta(channel, meta, ['title', 'description', 'isPublic'])
  56. // prepare channel category if needed
  57. if (isSet(meta.category)) {
  58. channel.category = await processChannelCategory(ctx, channel.category, parseInt(meta.category))
  59. }
  60. // prepare cover photo asset if needed
  61. if (isSet(meta.coverPhoto)) {
  62. const asset = findAssetByIndex(processedAssets, meta.coverPhoto, 'channel cover photo')
  63. if (asset) {
  64. channel.coverPhoto = asset
  65. }
  66. }
  67. // prepare avatar photo asset if needed
  68. if (isSet(meta.avatarPhoto)) {
  69. const asset = findAssetByIndex(processedAssets, meta.avatarPhoto, 'channel avatar photo')
  70. if (asset) {
  71. channel.avatarPhoto = asset
  72. }
  73. }
  74. // prepare language if needed
  75. if (isSet(meta.language)) {
  76. channel.language = await processLanguage(ctx, channel.language, meta.language)
  77. }
  78. return channel
  79. }
  80. export async function processVideoMetadata(
  81. ctx: EventContext & StoreContext,
  82. channel: Channel,
  83. video: Video,
  84. meta: DecodedMetadataObject<IVideoMetadata>,
  85. assets: NewAsset[]
  86. ): Promise<Video> {
  87. const assetsOwner = new DataObjectOwnerChannel()
  88. assetsOwner.channelId = channel.id
  89. const processedAssets = await Promise.all(assets.map((asset) => processNewAsset(ctx, asset, assetsOwner)))
  90. integrateMeta(video, meta, ['title', 'description', 'duration', 'hasMarketing', 'isExplicit', 'isPublic'])
  91. // prepare video category if needed
  92. if (meta.category) {
  93. video.category = await processVideoCategory(ctx, video.category, parseInt(meta.category))
  94. }
  95. // prepare media meta information if needed
  96. if (isSet(meta.mediaType) || isSet(meta.mediaPixelWidth) || isSet(meta.mediaPixelHeight)) {
  97. // prepare video file size if poosible
  98. const videoSize = extractVideoSize(assets, meta.video)
  99. video.mediaMetadata = await processVideoMediaMetadata(ctx, video.mediaMetadata, meta, videoSize)
  100. }
  101. // prepare license if needed
  102. if (isSet(meta.license)) {
  103. await updateVideoLicense(ctx, video, meta.license)
  104. }
  105. // prepare thumbnail photo asset if needed
  106. if (isSet(meta.thumbnailPhoto)) {
  107. const asset = findAssetByIndex(processedAssets, meta.thumbnailPhoto, 'thumbnail photo')
  108. if (asset) {
  109. video.thumbnailPhoto = asset
  110. }
  111. }
  112. // prepare video asset if needed
  113. if (isSet(meta.video)) {
  114. const asset = findAssetByIndex(processedAssets, meta.video, 'video')
  115. if (asset) {
  116. video.media = asset
  117. }
  118. }
  119. // prepare language if needed
  120. if (isSet(meta.language)) {
  121. video.language = await processLanguage(ctx, video.language, meta.language)
  122. }
  123. if (isSet(meta.publishedBeforeJoystream)) {
  124. video.publishedBeforeJoystream = processPublishedBeforeJoystream(
  125. ctx,
  126. video.publishedBeforeJoystream,
  127. meta.publishedBeforeJoystream
  128. )
  129. }
  130. return video
  131. }
  132. function findAssetByIndex(assets: typeof Asset[], index: number, name?: string): typeof Asset | null {
  133. if (assets[index]) {
  134. return assets[index]
  135. } else {
  136. invalidMetadata(`Invalid${name ? ' ' + name : ''} asset index`, {
  137. numberOfAssets: assets.length,
  138. requestedAssetIndex: index,
  139. })
  140. return null
  141. }
  142. }
  143. async function processVideoMediaEncoding(
  144. { store, event }: StoreContext & EventContext,
  145. existingVideoMediaEncoding: VideoMediaEncoding | undefined,
  146. metadata: DecodedMetadataObject<IMediaType>
  147. ): Promise<VideoMediaEncoding> {
  148. const encoding =
  149. existingVideoMediaEncoding ||
  150. new VideoMediaEncoding({
  151. createdAt: new Date(event.blockTimestamp),
  152. createdById: '1',
  153. updatedById: '1',
  154. })
  155. // integrate media encoding-related data
  156. integrateMeta(encoding, metadata, ['codecName', 'container', 'mimeMediaType'])
  157. encoding.updatedAt = new Date(event.blockTimestamp)
  158. await store.save<VideoMediaEncoding>(encoding)
  159. return encoding
  160. }
  161. async function processVideoMediaMetadata(
  162. ctx: StoreContext & EventContext,
  163. existingVideoMedia: VideoMediaMetadata | undefined,
  164. metadata: DecodedMetadataObject<IVideoMetadata>,
  165. videoSize: number | undefined
  166. ): Promise<VideoMediaMetadata> {
  167. const { store, event } = ctx
  168. const videoMedia =
  169. existingVideoMedia ||
  170. new VideoMediaMetadata({
  171. createdInBlock: event.blockNumber,
  172. createdAt: new Date(event.blockTimestamp),
  173. createdById: '1',
  174. updatedById: '1',
  175. })
  176. // integrate media-related data
  177. const mediaMetadata = {
  178. size: isSet(videoSize) ? new BN(videoSize.toString()) : undefined,
  179. pixelWidth: metadata.mediaPixelWidth,
  180. pixelHeight: metadata.mediaPixelHeight,
  181. }
  182. integrateMeta(videoMedia, mediaMetadata, ['pixelWidth', 'pixelHeight', 'size'])
  183. videoMedia.updatedAt = new Date(event.blockTimestamp)
  184. videoMedia.encoding = await processVideoMediaEncoding(ctx, videoMedia.encoding, metadata.mediaType || {})
  185. await store.save<VideoMediaMetadata>(videoMedia)
  186. return videoMedia
  187. }
  188. export async function convertContentActorToChannelOwner(
  189. store: DatabaseManager,
  190. contentActor: ContentActor
  191. ): Promise<{
  192. ownerMember?: Membership
  193. ownerCuratorGroup?: CuratorGroup
  194. }> {
  195. if (contentActor.isMember) {
  196. const memberId = contentActor.asMember.toNumber()
  197. const member = await store.get(Membership, { where: { id: memberId.toString() } as FindConditions<Membership> })
  198. // ensure member exists
  199. if (!member) {
  200. return inconsistentState(`Actor is non-existing member`, memberId)
  201. }
  202. return {
  203. ownerMember: member,
  204. ownerCuratorGroup: undefined, // this will clear the field
  205. }
  206. }
  207. if (contentActor.isCurator) {
  208. const curatorGroupId = contentActor.asCurator[0].toNumber()
  209. const curatorGroup = await store.get(CuratorGroup, {
  210. where: { id: curatorGroupId.toString() } as FindConditions<CuratorGroup>,
  211. })
  212. // ensure curator group exists
  213. if (!curatorGroup) {
  214. return inconsistentState('Actor is non-existing curator group', curatorGroupId)
  215. }
  216. return {
  217. ownerMember: undefined, // this will clear the field
  218. ownerCuratorGroup: curatorGroup,
  219. }
  220. }
  221. // TODO: contentActor.isLead
  222. logger.error('Not implemented ContentActor type', { contentActor: contentActor.toString() })
  223. throw new Error('Not-implemented ContentActor type used')
  224. }
  225. function processPublishedBeforeJoystream(
  226. ctx: EventContext & StoreContext,
  227. currentValue: Date | undefined,
  228. metadata: DecodedMetadataObject<IPublishedBeforeJoystream>
  229. ): Date | undefined {
  230. if (!isSet(metadata)) {
  231. return currentValue
  232. }
  233. // Property is beeing unset
  234. if (!metadata.isPublished) {
  235. return undefined
  236. }
  237. // try to parse timestamp from publish date
  238. const timestamp = isSet(metadata.date) ? Date.parse(metadata.date) : NaN
  239. // ensure date is valid
  240. if (isNaN(timestamp)) {
  241. invalidMetadata(`Invalid date used for publishedBeforeJoystream`, {
  242. timestamp,
  243. })
  244. return currentValue
  245. }
  246. // set new date
  247. return new Date(timestamp)
  248. }
  249. async function processNewAsset(
  250. ctx: EventContext & StoreContext,
  251. asset: NewAsset,
  252. owner: typeof DataObjectOwner
  253. ): Promise<typeof Asset> {
  254. if (asset.isUrls) {
  255. const urls = asset.asUrls.toArray().map((url) => url.toString())
  256. const resultAsset = new AssetExternal()
  257. resultAsset.urls = JSON.stringify(urls)
  258. return resultAsset
  259. } else if (asset.isUpload) {
  260. const contentParameters: ContentParameters = asset.asUpload
  261. const dataObject = await createDataObject(ctx, contentParameters, owner)
  262. const resultAsset = new AssetJoystreamStorage()
  263. resultAsset.dataObjectId = dataObject.id
  264. return resultAsset
  265. } else {
  266. unexpectedData('Unrecognized asset type', asset.type)
  267. }
  268. }
  269. function extractVideoSize(assets: NewAsset[], assetIndex: number | null | undefined): number | undefined {
  270. // escape if no asset is required
  271. if (!isSet(assetIndex)) {
  272. return undefined
  273. }
  274. // ensure asset index is valid
  275. if (assetIndex > assets.length) {
  276. invalidMetadata(`Non-existing asset video size extraction requested`, { assetsProvided: assets.length, assetIndex })
  277. return undefined
  278. }
  279. const rawAsset = assets[assetIndex]
  280. // escape if asset is describing URLs (can't get size)
  281. if (rawAsset.isUrls) {
  282. return undefined
  283. }
  284. // !rawAsset.isUrls && rawAsset.isUpload // asset is in storage
  285. // convert generic content parameters coming from processor to custom Joystream data type
  286. const customContentParameters = new Custom_ContentParameters(registry, rawAsset.asUpload.toJSON() as any)
  287. // extract video size
  288. const videoSize = customContentParameters.size_in_bytes.toNumber()
  289. return videoSize
  290. }
  291. async function processLanguage(
  292. ctx: EventContext & StoreContext,
  293. currentLanguage: Language | undefined,
  294. languageIso: string | undefined
  295. ): Promise<Language | undefined> {
  296. const { event, store } = ctx
  297. if (!isSet(languageIso)) {
  298. return currentLanguage
  299. }
  300. // validate language string
  301. const isValidIso = ISO6391.validate(languageIso)
  302. // ensure language string is valid
  303. if (!isValidIso) {
  304. invalidMetadata(`Invalid language ISO-639-1 provided`, languageIso)
  305. return currentLanguage
  306. }
  307. // load language
  308. const existingLanguage = await store.get(Language, { where: { iso: languageIso } })
  309. // return existing language if any
  310. if (existingLanguage) {
  311. return existingLanguage
  312. }
  313. // create new language
  314. const newLanguage = new Language({
  315. iso: languageIso,
  316. createdInBlock: event.blockNumber,
  317. createdAt: new Date(event.blockTimestamp),
  318. updatedAt: new Date(event.blockTimestamp),
  319. // TODO: remove these lines after Hydra auto-fills the values when cascading save (remove them on all places)
  320. createdById: '1',
  321. updatedById: '1',
  322. })
  323. await store.save<Language>(newLanguage)
  324. return newLanguage
  325. }
  326. async function updateVideoLicense(
  327. ctx: StoreContext & EventContext,
  328. video: Video,
  329. licenseMetadata: ILicense | null | undefined
  330. ): Promise<void> {
  331. const { store, event } = ctx
  332. if (!isSet(licenseMetadata)) {
  333. return
  334. }
  335. const previousLicense = video.license
  336. let license: License | null = null
  337. if (!isLicenseEmpty(licenseMetadata)) {
  338. // license is meant to be created/updated
  339. license =
  340. previousLicense ||
  341. new License({
  342. createdAt: new Date(event.blockTimestamp),
  343. createdById: '1',
  344. updatedById: '1',
  345. })
  346. license.updatedAt = new Date(event.blockTimestamp)
  347. integrateMeta(license, licenseMetadata, ['attribution', 'code', 'customText'])
  348. await store.save<License>(license)
  349. }
  350. // Update license (and potentially remove foreign key reference)
  351. // FIXME: Note that we MUST to provide "null" here in order to unset a relation,
  352. // See: https://github.com/Joystream/hydra/issues/435
  353. video.license = license as License | undefined
  354. video.updatedAt = new Date(ctx.event.blockTimestamp)
  355. await store.save<Video>(video)
  356. // Safely remove previous license if needed
  357. if (previousLicense && !license) {
  358. await store.remove<License>(previousLicense)
  359. }
  360. }
  361. /*
  362. Checks if protobof contains license with some fields filled or is empty object (`{}` or `{someKey: undefined, ...}`).
  363. Empty object means deletion is requested.
  364. */
  365. function isLicenseEmpty(licenseObject: ILicense): boolean {
  366. const somePropertySet = Object.values(licenseObject).some((v) => isSet(v))
  367. return !somePropertySet
  368. }
  369. async function processVideoCategory(
  370. ctx: EventContext & StoreContext,
  371. currentCategory: VideoCategory | undefined,
  372. categoryId: number
  373. ): Promise<VideoCategory | undefined> {
  374. const { store } = ctx
  375. // load video category
  376. const category = await store.get(VideoCategory, {
  377. where: { id: categoryId.toString() },
  378. })
  379. // ensure video category exists
  380. if (!category) {
  381. invalidMetadata('Non-existing video category association with video requested', categoryId)
  382. return currentCategory
  383. }
  384. return category
  385. }
  386. async function processChannelCategory(
  387. ctx: EventContext & StoreContext,
  388. currentCategory: ChannelCategory | undefined,
  389. categoryId: number
  390. ): Promise<ChannelCategory | undefined> {
  391. const { store } = ctx
  392. // load video category
  393. const category = await store.get(ChannelCategory, {
  394. where: { id: categoryId.toString() },
  395. })
  396. // ensure video category exists
  397. if (!category) {
  398. invalidMetadata('Non-existing channel category association with channel requested', categoryId)
  399. return currentCategory
  400. }
  401. return category
  402. }