utils.ts 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521
  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. }
  124. invalidMetadata(`Invalid${name ? ' ' + name : ''} asset index`, {
  125. numberOfAssets: assets.length,
  126. requestedAssetIndex: index,
  127. })
  128. return null
  129. }
  130. async function processVideoMediaEncoding(
  131. { store, event }: StoreContext & EventContext,
  132. existingVideoMediaEncoding: VideoMediaEncoding | undefined,
  133. metadata: DecodedMetadataObject<IMediaType>
  134. ): Promise<VideoMediaEncoding> {
  135. const encoding =
  136. existingVideoMediaEncoding ||
  137. new VideoMediaEncoding({
  138. createdAt: new Date(event.blockTimestamp),
  139. createdById: '1',
  140. updatedById: '1',
  141. })
  142. // integrate media encoding-related data
  143. integrateMeta(encoding, metadata, ['codecName', 'container', 'mimeMediaType'])
  144. encoding.updatedAt = new Date(event.blockTimestamp)
  145. await store.save<VideoMediaEncoding>(encoding)
  146. return encoding
  147. }
  148. async function processVideoMediaMetadata(
  149. ctx: StoreContext & EventContext,
  150. existingVideoMedia: VideoMediaMetadata | undefined,
  151. metadata: DecodedMetadataObject<IVideoMetadata>,
  152. videoSize: number | undefined
  153. ): Promise<VideoMediaMetadata> {
  154. const { store, event } = ctx
  155. const videoMedia =
  156. existingVideoMedia ||
  157. new VideoMediaMetadata({
  158. createdInBlock: event.blockNumber,
  159. createdAt: new Date(event.blockTimestamp),
  160. createdById: '1',
  161. updatedById: '1',
  162. })
  163. // integrate media-related data
  164. const mediaMetadata = {
  165. size: isSet(videoSize) ? new BN(videoSize.toString()) : undefined,
  166. pixelWidth: metadata.mediaPixelWidth,
  167. pixelHeight: metadata.mediaPixelHeight,
  168. }
  169. integrateMeta(videoMedia, mediaMetadata, ['pixelWidth', 'pixelHeight', 'size'])
  170. videoMedia.updatedAt = new Date(event.blockTimestamp)
  171. videoMedia.encoding = await processVideoMediaEncoding(ctx, videoMedia.encoding, metadata.mediaType || {})
  172. await store.save<VideoMediaMetadata>(videoMedia)
  173. return videoMedia
  174. }
  175. export async function convertContentActorToChannelOwner(
  176. store: DatabaseManager,
  177. contentActor: ContentActor
  178. ): Promise<{
  179. ownerMember?: Membership
  180. ownerCuratorGroup?: CuratorGroup
  181. }> {
  182. if (contentActor.isMember) {
  183. const memberId = contentActor.asMember.toNumber()
  184. const member = await store.get(Membership, { where: { id: memberId.toString() } as FindConditions<Membership> })
  185. // ensure member exists
  186. if (!member) {
  187. return inconsistentState(`Actor is non-existing member`, memberId)
  188. }
  189. return {
  190. ownerMember: member,
  191. ownerCuratorGroup: undefined, // this will clear the field
  192. }
  193. }
  194. if (contentActor.isCurator) {
  195. const curatorGroupId = contentActor.asCurator[0].toNumber()
  196. const curatorGroup = await store.get(CuratorGroup, {
  197. where: { id: curatorGroupId.toString() } as FindConditions<CuratorGroup>,
  198. })
  199. // ensure curator group exists
  200. if (!curatorGroup) {
  201. return inconsistentState('Actor is non-existing curator group', curatorGroupId)
  202. }
  203. return {
  204. ownerMember: undefined, // this will clear the field
  205. ownerCuratorGroup: curatorGroup,
  206. }
  207. }
  208. // TODO: contentActor.isLead
  209. logger.error('Not implemented ContentActor type', { contentActor: contentActor.toString() })
  210. throw new Error('Not-implemented ContentActor type used')
  211. }
  212. function processPublishedBeforeJoystream(
  213. ctx: EventContext & StoreContext,
  214. currentValue: Date | undefined,
  215. metadata: DecodedMetadataObject<IPublishedBeforeJoystream>
  216. ): Date | undefined {
  217. if (!isSet(metadata)) {
  218. return currentValue
  219. }
  220. // Property is beeing unset
  221. if (!metadata.isPublished) {
  222. return undefined
  223. }
  224. // try to parse timestamp from publish date
  225. const timestamp = isSet(metadata.date) ? Date.parse(metadata.date) : NaN
  226. // ensure date is valid
  227. if (isNaN(timestamp)) {
  228. invalidMetadata(`Invalid date used for publishedBeforeJoystream`, {
  229. timestamp,
  230. })
  231. return currentValue
  232. }
  233. // set new date
  234. return new Date(timestamp)
  235. }
  236. async function processNewAssets(ctx: EventContext & StoreContext, assets: NewAssets): Promise<Array<typeof Asset>> {
  237. if (assets.isUrls) {
  238. return assets.asUrls.map((assetUrls) => {
  239. const resultAsset = new AssetExternal()
  240. resultAsset.urls = JSON.stringify(assetUrls.map((u) => u.toString()))
  241. return resultAsset
  242. })
  243. }
  244. 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. }
  254. unexpectedData('Unrecognized assets type', assets.type)
  255. }
  256. function extractVideoSize(assets: NewAssets | undefined, assetIndex: number | null | undefined): number | undefined {
  257. // escape if no assetIndex is set
  258. if (!isSet(assetIndex)) {
  259. return undefined
  260. }
  261. // index provided, but there are no assets
  262. if (!assets) {
  263. invalidMetadata(`Non-existing asset video size extraction requested - no assets were uploaded!`, {
  264. assetIndex,
  265. })
  266. return undefined
  267. }
  268. // cannot extract size from other asset types than "Upload"
  269. if (!assets.isUpload) {
  270. return undefined
  271. }
  272. const dataObjectsParams = assets.asUpload.object_creation_list
  273. // ensure asset index is valid
  274. if (assetIndex >= dataObjectsParams.length) {
  275. invalidMetadata(`Non-existing asset video size extraction requested`, {
  276. assetsProvided: dataObjectsParams.length,
  277. assetIndex,
  278. })
  279. return undefined
  280. }
  281. // extract video size from objectParams
  282. const objectParams = assets.asUpload.object_creation_list[assetIndex]
  283. const params = new ObjectCreationParams(registry, objectParams.toJSON() as any)
  284. const videoSize = params.getField('size').toNumber()
  285. return videoSize
  286. }
  287. async function processLanguage(
  288. ctx: EventContext & StoreContext,
  289. currentLanguage: Language | undefined,
  290. languageIso: string | undefined
  291. ): Promise<Language | undefined> {
  292. const { event, store } = ctx
  293. if (!isSet(languageIso)) {
  294. return currentLanguage
  295. }
  296. // ensure language string is valid
  297. if (!isValidLanguageCode(languageIso)) {
  298. invalidMetadata(`Invalid language ISO-639-1 provided`, languageIso)
  299. return currentLanguage
  300. }
  301. // load language
  302. const existingLanguage = await store.get(Language, { where: { iso: languageIso } })
  303. // return existing language if any
  304. if (existingLanguage) {
  305. return existingLanguage
  306. }
  307. // create new language
  308. const newLanguage = new Language({
  309. iso: languageIso,
  310. createdInBlock: event.blockNumber,
  311. createdAt: new Date(event.blockTimestamp),
  312. updatedAt: new Date(event.blockTimestamp),
  313. // TODO: remove these lines after Hydra auto-fills the values when cascading save (remove them on all places)
  314. createdById: '1',
  315. updatedById: '1',
  316. })
  317. await store.save<Language>(newLanguage)
  318. return newLanguage
  319. }
  320. async function updateVideoLicense(
  321. ctx: StoreContext & EventContext,
  322. video: Video,
  323. licenseMetadata: ILicense | null | undefined
  324. ): Promise<void> {
  325. const { store, event } = ctx
  326. if (!isSet(licenseMetadata)) {
  327. return
  328. }
  329. const previousLicense = video.license
  330. let license: License | null = null
  331. if (!isLicenseEmpty(licenseMetadata)) {
  332. // license is meant to be created/updated
  333. license =
  334. previousLicense ||
  335. new License({
  336. createdAt: new Date(event.blockTimestamp),
  337. createdById: '1',
  338. updatedById: '1',
  339. })
  340. license.updatedAt = new Date(event.blockTimestamp)
  341. integrateMeta(license, licenseMetadata, ['attribution', 'code', 'customText'])
  342. await store.save<License>(license)
  343. }
  344. // Update license (and potentially remove foreign key reference)
  345. // FIXME: Note that we MUST to provide "null" here in order to unset a relation,
  346. // See: https://github.com/Joystream/hydra/issues/435
  347. video.license = license as License | undefined
  348. video.updatedAt = new Date(ctx.event.blockTimestamp)
  349. await store.save<Video>(video)
  350. // Safely remove previous license if needed
  351. if (previousLicense && !license) {
  352. await store.remove<License>(previousLicense)
  353. }
  354. }
  355. /*
  356. Checks if protobof contains license with some fields filled or is empty object (`{}` or `{someKey: undefined, ...}`).
  357. Empty object means deletion is requested.
  358. */
  359. function isLicenseEmpty(licenseObject: ILicense): boolean {
  360. const somePropertySet = Object.values(licenseObject).some((v) => isSet(v))
  361. return !somePropertySet
  362. }
  363. async function processVideoCategory(
  364. ctx: EventContext & StoreContext,
  365. currentCategory: VideoCategory | undefined,
  366. categoryId: number
  367. ): Promise<VideoCategory | undefined> {
  368. const { store } = ctx
  369. // load video category
  370. const category = await store.get(VideoCategory, {
  371. where: { id: categoryId.toString() },
  372. })
  373. // ensure video category exists
  374. if (!category) {
  375. invalidMetadata('Non-existing video category association with video requested', categoryId)
  376. return currentCategory
  377. }
  378. return category
  379. }
  380. async function processChannelCategory(
  381. ctx: EventContext & StoreContext,
  382. currentCategory: ChannelCategory | undefined,
  383. categoryId: number
  384. ): Promise<ChannelCategory | undefined> {
  385. const { store } = ctx
  386. // load video category
  387. const category = await store.get(ChannelCategory, {
  388. where: { id: categoryId.toString() },
  389. })
  390. // ensure video category exists
  391. if (!category) {
  392. invalidMetadata('Non-existing channel category association with channel requested', categoryId)
  393. return currentCategory
  394. }
  395. return category
  396. }
  397. // Needs to be done every time before data object is removed!
  398. export async function unsetAssetRelations(store: DatabaseManager, dataObject: StorageDataObject): Promise<void> {
  399. const channelAssets = ['avatarPhoto', 'coverPhoto'] as const
  400. const videoAssets = ['thumbnailPhoto', 'media'] as const
  401. // NOTE: we don't need to retrieve multiple channels/videos via `store.getMany()` because dataObject
  402. // is allowed to be associated only with one channel/video in runtime
  403. const channel = await store.get(Channel, {
  404. where: channelAssets.map((assetName) => ({
  405. [assetName]: Raw((alias) => `${alias} ->> 'dataObjectId' = :id`, {
  406. id: dataObject.id,
  407. }),
  408. })),
  409. })
  410. const video = await store.get(Video, {
  411. where: videoAssets.map((assetName) => ({
  412. [assetName]: Raw((alias) => `${alias} ->> 'dataObjectId' = :id`, {
  413. id: dataObject.id,
  414. }),
  415. })),
  416. })
  417. if (channel) {
  418. channelAssets.forEach((assetName) => {
  419. if (channel[assetName] && (channel[assetName] as AssetJoystreamStorage).dataObjectId === dataObject.id) {
  420. channel[assetName] = new AssetNone()
  421. }
  422. })
  423. await store.save<Channel>(channel)
  424. // emit log event
  425. logger.info('Content has been disconnected from Channel', {
  426. channelId: channel.id.toString(),
  427. dataObjectId: dataObject.id,
  428. })
  429. }
  430. 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. }