utils.ts 15 KB

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