From 7d8760460a9fa264646524e94bcace1c0e43d108 Mon Sep 17 00:00:00 2001
From: AuroraHuang22 <75730405+AuroraHuang22@users.noreply.github.com>
Date: Mon, 6 Nov 2023 16:58:35 +0800
Subject: [PATCH] =?UTF-8?q?=F0=9F=9A=B8=20Extract=20Epub=20metadata=20for?=
=?UTF-8?q?=20ISCN=20registration=20(#408)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* ✨ Get epub metadata & coverInfo
* ✨ Add epubMetadata to thumbnail field
* 🚸 Provide additional information for sameAs
* ✨ Add ISBN field for Book
* 🐛 Fix read fileType undefined
* 🚸 Halt further execution on error
* 🎨 Improve naming & Enhance type definitions
* 🎨 Improve logic for fetching cover file
* 🎨 Store arweaveId without adding prefix
* 🎨 Store epubMetadata after processing
* 🚸 Display only epub & pdf in sameAs
* ✏️ Rename formatUrlOptions to filteredUrlOptions
* 🥅 Guard empty arweaveId
---
components/IscnRegisterForm.vue | 38 +++++++-
components/IscnUploadForm.vue | 156 +++++++++++++++++++++++++++++---
components/SameAsFieldList.vue | 73 +++++++++++++--
constant/index.ts | 2 +-
locales/en.json | 3 +
package.json | 1 +
pages/new/index.vue | 7 ++
utils/cosmos/iscn/iscn.type.ts | 3 +
yarn.lock | 129 +++++++++++++++++++++++++-
9 files changed, 386 insertions(+), 26 deletions(-)
diff --git a/components/IscnRegisterForm.vue b/components/IscnRegisterForm.vue
index 9a7c19ba..c2230a32 100644
--- a/components/IscnRegisterForm.vue
+++ b/components/IscnRegisterForm.vue
@@ -355,6 +355,15 @@
:placeholder="$t('IscnRegisterForm.placeholder.url')"
/>
+
+
+
@@ -740,6 +750,7 @@ export enum AuthorDialogType {
export default class IscnRegisterForm extends Vue {
@Prop({ default: [] }) readonly fileRecords!: any[]
@Prop({ default: [] }) readonly uploadArweaveList!: string[]
+ @Prop() readonly epubMetadata!: any | null
@Prop(String) readonly ipfsHash!: string
@Prop(String) readonly arweaveId!: string
@@ -759,7 +770,7 @@ export default class IscnRegisterForm extends Vue {
fileTypeOptions = [
'epub',
'pdf',
- 'mp3',
+ 'audio',
'jpg',
'png',
]
@@ -784,8 +795,10 @@ export default class IscnRegisterForm extends Vue {
tags: string[] = []
sameAs: string[] = []
url: string = ''
+ isbn: string = ''
license: string = this.licenseOptions[0]
customLicense: string = ''
+ thumbnailUrl: string = ''
authorName: string = ''
authorUrl: string[] = []
authorWalletAddress: string[] = []
@@ -834,6 +847,7 @@ export default class IscnRegisterForm extends Vue {
currentAuthorDialogType: AuthorDialogType = AuthorDialogType.stakeholder
sameAsList: any = []
+ language: string = ''
get ipfsHashList() {
const list = []
@@ -987,6 +1001,7 @@ export default class IscnRegisterForm extends Vue {
tagsString: this.tagsString,
sameAs: this.formattedSameAsList,
url: this.url,
+ isbn: this.isbn,
exifInfo: this.exif.filter(file => file),
license: this.formattedLicense,
ipfsHash: this.ipfsHashList,
@@ -1001,6 +1016,8 @@ export default class IscnRegisterForm extends Vue {
likerIdsAddresses: this.likerIdsAddresses,
authorDescriptions: this.authorDescriptions,
contentFingerprints: this.customContentFingerprints,
+ inLanguage: this.language,
+ thumbnailUrl: this.thumbnailUrl,
}
}
@@ -1084,6 +1101,16 @@ export default class IscnRegisterForm extends Vue {
}
async mounted() {
+ if (this.epubMetadata) {
+ this.name = this.epubMetadata.title;
+ this.description = this.extractText(this.epubMetadata.description);
+ this.author.name = this.epubMetadata.author;
+ this.language = this.epubMetadata.language
+ this.tags = this.epubMetadata.tags
+ this.thumbnailUrl = this.formatArweave(this.epubMetadata.thumbnailUrl) as string
+ if (this.author.name) { this.authors.push(this.author) }
+ }
+
this.uploadStatus = 'loading'
// ISCN Fee needs Arweave fee to calculate
await this.calculateISCNFee()
@@ -1433,5 +1460,14 @@ export default class IscnRegisterForm extends Vue {
this.displayImageSrc = this.fileRecords[index].fileData
this.displayExifInfo = this.fileRecords[index].exifInfo
}
+
+ // eslint-disable-next-line class-methods-use-this
+ extractText(htmlString: string) {
+ if (!htmlString) return ''
+ const div = document.createElement('div');
+ div.innerHTML = htmlString;
+ div.innerHTML = div.innerHTML.replace(/
/gi, "\n");
+ return div.textContent || div.innerText;
+ }
}
diff --git a/components/IscnUploadForm.vue b/components/IscnUploadForm.vue
index 7ef8fea9..2f7a831f 100644
--- a/components/IscnUploadForm.vue
+++ b/components/IscnUploadForm.vue
@@ -227,6 +227,7 @@ import { namespace } from 'vuex-class'
import exifr from 'exifr'
import Hash from 'ipfs-only-hash'
import BigNumber from 'bignumber.js'
+import ePub from 'epubjs';
import { OfflineSigner } from '@cosmjs/proto-signing'
@@ -282,8 +283,6 @@ export default class UploadForm extends Vue {
string, { transactionHash?: string, arweaveId?: string }
>()
-
-
likerId: string = ''
error: string = ''
shouldShowAlert = false
@@ -292,6 +291,8 @@ export default class UploadForm extends Vue {
signDialogError = ''
balance = new BigNumber(0)
+ epubMetadataList: any[] = []
+
get formClasses() {
return [
'flex',
@@ -356,7 +357,7 @@ export default class UploadForm extends Vue {
case 'MISSING_SIGNER':
return this.$t('IscnRegisterForm.error.missingSigner')
default:
- return ''
+ return this.error
}
}
@@ -442,6 +443,10 @@ export default class UploadForm extends Vue {
console.error(err)
}
}
+ if (file.type === 'application/epub+zip') {
+ // eslint-disable-next-line no-await-in-loop
+ await this.processEPub({ buffer: fileBytes, file })
+ }
}
} else {
this.isSizeExceeded = true
@@ -451,6 +456,109 @@ export default class UploadForm extends Vue {
}
}
+ async processEPub({ buffer, file }: { buffer: ArrayBuffer; file: File }) {
+ try {
+ const book = ePub(buffer)
+ await book.ready
+ const epubMetadata: any = {}
+
+ // Get metadata
+ const { metadata } = book.packaging
+ if (metadata) {
+ epubMetadata.epubFileName = file.name
+ epubMetadata.title = metadata.title
+ epubMetadata.author = metadata.creator
+ epubMetadata.language = this.formatLanguage(metadata.language)
+ epubMetadata.description = metadata.description
+ }
+
+ // Get tags
+ const opfFilePath = await (book.path as any).path
+ const opfContent = await book.archive.getText(opfFilePath)
+ const parser = new DOMParser()
+ const opfDocument = parser.parseFromString(opfContent, 'application/xml')
+ const dcSubjectElements = opfDocument.querySelectorAll(
+ 'dc\\:subject, subject',
+ )
+ const subjects: string[] = []
+ dcSubjectElements.forEach((element) => {
+ const subject = element.textContent
+ if (subject) {
+ subjects.push(subject)
+ }
+ })
+ epubMetadata.tags = subjects
+
+ // Get cover file
+ const coverUrl = await book.coverUrl()
+ if (coverUrl) {
+ const response = await fetch(coverUrl)
+ const blobData = await response.blob()
+ if (blobData) {
+ const coverFile = new File(
+ [blobData],
+ `${metadata.title}_cover.jpeg`,
+ {
+ type: 'image/jpeg',
+ },
+ )
+ const fileBytes = (await fileToArrayBuffer(
+ coverFile,
+ )) as unknown as ArrayBuffer
+ if (fileBytes) {
+ const [
+ fileSHA256,
+ imageType,
+ ipfsHash,
+ // eslint-disable-next-line no-await-in-loop
+ ] = await Promise.all([
+ digestFileSHA256(fileBytes),
+ readImageType(fileBytes),
+ Hash.of(Buffer.from(fileBytes)),
+ ])
+
+ epubMetadata.ipfsHash = ipfsHash
+
+ const fileRecord: any = {
+ fileName: coverFile.name,
+ fileSize: coverFile.size,
+ fileType: coverFile.type,
+ fileBlob: coverFile,
+ ipfsHash,
+ fileSHA256,
+ isFileImage: !!imageType,
+ }
+ const reader = new FileReader()
+ reader.onload = (e) => {
+ if (!e.target) return
+ fileRecord.fileData = e.target.result as string
+ this.fileRecords.push(fileRecord)
+ }
+ reader.readAsDataURL(coverFile)
+ }
+ }
+ }
+ this.epubMetadataList.push(epubMetadata)
+ } catch (err) {
+ console.error(err)
+ }
+ }
+
+ // eslint-disable-next-line class-methods-use-this
+ formatLanguage(language: string) {
+ let formattedLanguage = '';
+ if (language) {
+ if (language.toLowerCase().startsWith('en')) {
+ formattedLanguage = 'en'
+ } else if (language.toLowerCase().startsWith('zh')) {
+ formattedLanguage = 'zh'
+ } else {
+ formattedLanguage = language
+ }
+ }
+ return formattedLanguage
+ }
+
onEnterURL() {
if (
!(
@@ -472,7 +580,13 @@ export default class UploadForm extends Vue {
}
handleDeleteFile(index: number) {
- this.fileRecords.splice(index, 1)
+ const deletedFile = this.fileRecords[index];
+ this.fileRecords.splice(index, 1);
+
+ const indexToDelete = this.epubMetadataList.findIndex(item => item.ipfsHash === deletedFile.ipfsHashList);
+ if (indexToDelete !== -1) {
+ this.epubMetadataList.splice(indexToDelete, 1);
+ }
}
handleClickExifInfo(index: number) {
@@ -514,6 +628,10 @@ export default class UploadForm extends Vue {
}
if (arweaveId) {
this.sentArweaveTransactionInfo.set(ipfsHash, { transactionHash: '', arweaveId });
+ const metadata = this.epubMetadataList.find((data: any) => data.ipfsHash === ipfsHash)
+ if (metadata) {
+ metadata.thumbnailUrl = arweaveId;
+ }
}
if (!this.arweaveFeeTargetAddress) {
this.arweaveFeeTargetAddress = address;
@@ -589,19 +707,21 @@ export default class UploadForm extends Vue {
if (arweaveId) {
const uploadedData = this.sentArweaveTransactionInfo.get(records.ipfsHash) || {};
this.sentArweaveTransactionInfo.set(records.ipfsHash, { ...uploadedData, arweaveId });
+ if (tempRecord.fileName.includes('cover.jpeg')) {
+ const metadata = this.epubMetadataList.find((file: any) => file.ipfsHash === records.ipfsHash)
+ metadata.thumbnailUrl = arweaveId
+ }
this.$emit('arweaveUploaded', { arweaveId })
this.isOpenSignDialog = false
} else {
- this.shouldShowAlert = true
- this.errorMessage = this.$t('IscnRegisterForm.error.arweave') as string
- this.$emit('handleContinue')
+ this.isOpenWarningSnackbar = true
+ this.error = this.$t('IscnRegisterForm.error.arweave') as string
+ throw new Error(this.error)
}
} catch (err) {
// TODO: Handle error
// eslint-disable-next-line no-console
- console.error(err)
- this.shouldShowAlert = true
- this.errorMessage = (err as Error).toString()
+ throw new Error(err as string)
}
}
@@ -637,12 +757,26 @@ export default class UploadForm extends Vue {
} catch (error) {
// eslint-disable-next-line no-console
console.error(error)
+ this.isOpenWarningSnackbar = true
+ this.error = (error as Error).toString()
+ this.uploadStatus = '';
+ return
} finally {
this.uploadStatus = '';
}
const uploadArweaveIdList = Array.from(this.sentArweaveTransactionInfo.values()).map(entry => entry.arweaveId);
- this.$emit('submit', { fileRecords: this.fileRecords, arweaveIds: uploadArweaveIdList })
+ this.fileRecords.forEach((record: any, index:number) => {
+ if (this.sentArweaveTransactionInfo.has(record.ipfsHash)) {
+ const arweaveId = this.sentArweaveTransactionInfo.get(
+ record.ipfsHash,
+ )?.arweaveId
+ if (arweaveId) {
+ this.fileRecords[index].arweaveId = arweaveId
+ }
+ }
+ })
+ this.$emit('submit', { fileRecords: this.fileRecords, arweaveIds: uploadArweaveIdList, epubMetadata: this.epubMetadataList[0] })
}
handleSignDialogClose() {
diff --git a/components/SameAsFieldList.vue b/components/SameAsFieldList.vue
index ce882d65..73a56370 100644
--- a/components/SameAsFieldList.vue
+++ b/components/SameAsFieldList.vue
@@ -16,13 +16,13 @@
-
+
-
+
handleSelectValue({ value, index: i })"
/>
+
@Prop({ default: () => [] }) readonly currentList!: Array
+ @Prop({ default: () => [] }) readonly fileRecords!: Array
@Prop(String) readonly name!: string | undefined
sameAsList: any = [{
@@ -86,6 +94,11 @@ export default class WalletFieldList extends Vue {
filetype: SAME_AS_FILE_TYPES[0],
}]
+ fileTypeToFind = [
+ 'epub',
+ 'pdf',
+ ]
+
// eslint-disable-next-line class-methods-use-this
get sameAsFiletypeOptions() {
return SAME_AS_FILE_TYPES
@@ -101,7 +114,7 @@ export default class WalletFieldList extends Vue {
return this.name.replace(/[.,!?;:'"(){}[\]<>]/g, '').replace(/\s+/g, '_').toUpperCase();
}
- get formatUrlOptions() {
+ get filteredUrlOptions() {
return this.urlOptions.filter(url => url.startsWith('ar://'))
}
@@ -112,20 +125,30 @@ export default class WalletFieldList extends Vue {
id: `${list.url}-${list.filename}`,
filename: list.filename,
filetype: list.filetype || SAME_AS_FILE_TYPES[0],
+ originFileName: list.originFileName,
}))
- } else if (this.formatUrlOptions.length) {
- this.sameAsList = this.formatUrlOptions.map((url, index) => ({
- url,
- id: `${url}-${index}`,
- filename: this.formatName,
- filetype: SAME_AS_FILE_TYPES[0],
- }))
+ } else if (this.filteredUrlOptions.length) {
+ this.sameAsList = this.fileRecords
+ .filter((file) => this.fileTypeToFind.includes(this.formatFileType(file.fileType)) && file.arweaveId)
+ .map((file, index) => {
+ const url = this.filteredUrlOptions.find((ar) => ar.includes(file.arweaveId));
+ const formattedFileType = this.formatFileType(file.fileType);
+
+ return {
+ url,
+ id: `${url}-${index}`,
+ filename: this.formatName,
+ filetype: formattedFileType || SAME_AS_FILE_TYPES[0],
+ originFileName: file.fileName || '',
+ };
+ });
} else {
this.sameAsList = [{
url: '',
id: 1,
filename: this.formatName,
filetype: SAME_AS_FILE_TYPES[0],
+ originFileName:'',
}]
}
}
@@ -159,5 +182,35 @@ export default class WalletFieldList extends Vue {
this.deleteEmptyField()
this.$emit('onConfirm', this.sameAsList)
}
+
+ // eslint-disable-next-line class-methods-use-this
+ formatFileType(fileType: string) {
+ let formattedFileType = ''
+ if (fileType) {
+ switch (true) {
+ case fileType.includes('jpg'):
+ case fileType.includes('jpeg'):
+ formattedFileType = 'jpg'
+ break
+ case fileType.includes('png'):
+ formattedFileType = 'png'
+ break
+ case fileType.includes('audio'):
+ // audio/ogg
+ formattedFileType = 'audio'
+ break
+ case fileType.includes('pdf'):
+ formattedFileType = 'pdf'
+ break
+ case fileType.includes('epub'):
+ formattedFileType = 'epub'
+ break
+ default:
+ formattedFileType = ''
+ break
+ }
+ }
+ return formattedFileType
+ }
}
diff --git a/constant/index.ts b/constant/index.ts
index a09ffea7..c62f91c9 100644
--- a/constant/index.ts
+++ b/constant/index.ts
@@ -62,7 +62,7 @@ export const WALLET_TYPES = [
export const SAME_AS_FILE_TYPES = [
'epub',
'pdf',
- 'mp3',
+ 'audio',
'jpg',
'png',
]
diff --git a/locales/en.json b/locales/en.json
index 39698d51..db70e273 100644
--- a/locales/en.json
+++ b/locales/en.json
@@ -214,12 +214,14 @@
"IscnRegisterForm.label.emptyFile": "No Content",
"IscnRegisterForm.label.fingerprints": "Content Fingerprints",
"IscnRegisterForm.label.iscn": "ISCN Title",
+ "IscnRegisterForm.label.isbn": "ISBN",
"IscnRegisterForm.label.license": "License",
"IscnRegisterForm.label.likerID": "LikerID",
"IscnRegisterForm.label.sameAs" : "Same As URLs",
"IscnRegisterForm.label.name": "Name",
"IscnRegisterForm.label.fileName": "File name",
"IscnRegisterForm.label.fileType": "File type",
+ "IscnRegisterForm.label.originFile": "origin file: {name}",
"IscnRegisterForm.label.numbersProtocol": "Numbers Protocol",
"IscnRegisterForm.label.numbersProtocol.details": "Register your image asset in {link}",
"IscnRegisterForm.label.numbersProtocol.details.link": "Numbers Protocol",
@@ -238,6 +240,7 @@
"IscnRegisterForm.placeholder.fileName": "File Name (en)",
"IscnRegisterForm.placeholder.fileType": "File Type",
"IscnRegisterForm.placeholder.url": "URL",
+ "IscnRegisterForm.placeholder.isbn": "ISBN 10/ ISBN 13",
"IscnRegisterForm.placeholder.wallet": "Wallet Address",
"IscnRegisterForm.quitAlertDialog.confirm": "Cancel Registration",
"IscnRegisterForm.quitAlertDialog.content": "Your registration will be cancelled if you leave this page, and the paid fee cannot be refunded.",
diff --git a/package.json b/package.json
index 6fc46c55..0229f7a3 100644
--- a/package.json
+++ b/package.json
@@ -40,6 +40,7 @@
"core-js": "^3.15.1",
"cosmjs-types": "^0.8.0",
"easyqrcodejs": "^4.4.6",
+ "epubjs": "^0.3.93",
"exifr": "^7.1.2",
"express": "^4.17.1",
"fast-json-stable-stringify": "^2.1.0",
diff --git a/pages/new/index.vue b/pages/new/index.vue
index 7abe7099..a71427e7 100644
--- a/pages/new/index.vue
+++ b/pages/new/index.vue
@@ -41,6 +41,7 @@
:ipfs-hash="urlIpfsHash"
:arweave-id="urlArweaveId"
:upload-arweave-list=uploadArweaveList
+ :epub-metadata="epubMetadata"
:step="step"
@txBroadcasted="onISCNTxInfo"
@@ -134,6 +135,7 @@ export default class NewIndexPage extends Vue {
uploadFileRecords: any[] = []
urlFileRecords: any[] = []
+ epubMetadata: any | null = null
get shouldSkipToMintNFT(): boolean {
return this.$route.query.mint === '1'
@@ -189,9 +191,11 @@ export default class NewIndexPage extends Vue {
onSubmitUpload({
fileRecords,
arweaveIds,
+ epubMetadata,
}: {
fileRecords: any[]
arweaveIds: string[]
+ epubMetadata: any
}) {
if (fileRecords && fileRecords.length) {
this.uploadFileRecords = [...fileRecords]
@@ -199,6 +203,9 @@ export default class NewIndexPage extends Vue {
if (arweaveIds && arweaveIds.length) {
this.uploadArweaveList = [...arweaveIds]
}
+ if (epubMetadata) {
+ this.epubMetadata = {...epubMetadata}
+ }
this.state = 'iscn'
logTrackerEvent(this, 'ISCNCreate', 'ISCNConfirmFile', '', 1)
}
diff --git a/utils/cosmos/iscn/iscn.type.ts b/utils/cosmos/iscn/iscn.type.ts
index d450ead7..7862b9ac 100644
--- a/utils/cosmos/iscn/iscn.type.ts
+++ b/utils/cosmos/iscn/iscn.type.ts
@@ -24,6 +24,9 @@ export interface ISCNRegisterPayload {
stakeholders?: any[];
recordNotes?: string;
memo?: string;
+ inLanguage?: string;
+ thumbnailUrl?: string;
+ isbn?: string;
}
export interface ISCNRecordWithID extends ISCNRecord {
id: string;
diff --git a/yarn.lock b/yarn.lock
index d592e15a..59ec1d7e 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -4970,6 +4970,13 @@
resolved "https://registry.yarnpkg.com/@types/less/-/less-3.0.2.tgz#2761d477678c8374cb9897666871662eb1d1115e"
integrity sha512-62vfe65cMSzYaWmpmhqCMMNl0khen89w57mByPi1OseGfcV/LV03fO8YVrNj7rFQsRWNJo650WWyh6m7p8vZmA==
+"@types/localforage@0.0.34":
+ version "0.0.34"
+ resolved "https://registry.yarnpkg.com/@types/localforage/-/localforage-0.0.34.tgz#5e31c32dd8791ec4b9ff3ef47c9cb55b2d0d9438"
+ integrity sha512-tJxahnjm9dEI1X+hQSC5f2BSd/coZaqbIl1m3TCl0q9SVuC52XcXfV0XmoCU1+PmjyucuVITwoTnN8OlTbEXXA==
+ dependencies:
+ localforage "*"
+
"@types/lodash.chunk@^4.2.6":
version "4.2.6"
resolved "https://registry.yarnpkg.com/@types/lodash.chunk/-/lodash.chunk-4.2.6.tgz#9d35f05360b0298715d7f3d9efb34dd4f77e5d2a"
@@ -6226,6 +6233,11 @@
"@webassemblyjs/wast-parser" "1.9.0"
"@xtuc/long" "4.2.2"
+"@xmldom/xmldom@^0.7.5":
+ version "0.7.13"
+ resolved "https://registry.yarnpkg.com/@xmldom/xmldom/-/xmldom-0.7.13.tgz#ff34942667a4e19a9f4a0996a76814daac364cf3"
+ integrity sha512-lm2GW5PkosIzccsaZIz7tp8cPADSIlIHWDFTR1N0SzfinhhYgeIQjFMz4rYzanCScr3DqQLeomUDArp6MWKm+g==
+
"@xtuc/ieee754@^1.2.0":
version "1.2.0"
resolved "https://registry.yarnpkg.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz#eef014a3145ae477a1cbc00cd1e552336dceb790"
@@ -8623,6 +8635,11 @@ core-js@^3.15.1:
resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.15.1.tgz#6c08ab88abdf56545045ccf5fd81f47f407e7f1a"
integrity sha512-h8VbZYnc9pDzueiS2610IULDkpFFPunHwIpl8yRwFahAEEdSpHlTy3h3z3rKq5h11CaUdBEeRViu9AYvbxiMeg==
+core-js@^3.18.3:
+ version "3.33.2"
+ resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.33.2.tgz#312bbf6996a3a517c04c99b9909cdd27138d1ceb"
+ integrity sha512-XeBzWI6QL3nJQiHmdzbAOiMYqjrb7hwU7A39Qhvd/POSa/t9E1AeZyEZx3fNvp/vtM8zXwhoL0FsiS0hD0pruQ==
+
core-util-is@~1.0.0:
version "1.0.2"
resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7"
@@ -9179,6 +9196,14 @@ cyclist@^1.0.1:
resolved "https://registry.yarnpkg.com/cyclist/-/cyclist-1.0.1.tgz#596e9698fd0c80e12038c2b82d6eb1b35b6224d9"
integrity sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk=
+d@1, d@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/d/-/d-1.0.1.tgz#8698095372d58dbee346ffd0c7093f99f8f9eb5a"
+ integrity sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==
+ dependencies:
+ es5-ext "^0.10.50"
+ type "^1.0.1"
+
data-urls@^2.0.0:
version "2.0.0"
resolved "https://registry.yarnpkg.com/data-urls/-/data-urls-2.0.0.tgz#156485a72963a970f5d5821aaf642bef2bf2db9b"
@@ -9885,6 +9910,21 @@ entities@^4.2.0, entities@^4.3.0:
resolved "https://registry.yarnpkg.com/entities/-/entities-4.3.0.tgz#62915f08d67353bb4eb67e3d62641a4059aec656"
integrity sha512-/iP1rZrSEJ0DTlPiX+jbzlA3eVkY/e8L8SozroF395fIqE3TYF/Nz7YOMAawta+vLmyJ/hkGNNPcSbMADCCXbg==
+epubjs@^0.3.93:
+ version "0.3.93"
+ resolved "https://registry.yarnpkg.com/epubjs/-/epubjs-0.3.93.tgz#100c4597db152fc07d5246be38acca928b6b0b22"
+ integrity sha512-c06pNSdBxcXv3dZSbXAVLE1/pmleRhOT6mXNZo6INKmvuKpYB65MwU/lO7830czCtjIiK9i+KR+3S+p0wtljrw==
+ dependencies:
+ "@types/localforage" "0.0.34"
+ "@xmldom/xmldom" "^0.7.5"
+ core-js "^3.18.3"
+ event-emitter "^0.3.5"
+ jszip "^3.7.1"
+ localforage "^1.10.0"
+ lodash "^4.17.21"
+ marks-pane "^1.0.9"
+ path-webpack "0.0.3"
+
err-code@^3.0.0, err-code@^3.0.1:
version "3.0.1"
resolved "https://registry.yarnpkg.com/err-code/-/err-code-3.0.1.tgz#a444c7b992705f2b120ee320b09972eef331c920"
@@ -9984,11 +10024,29 @@ es-to-primitive@^1.2.1:
is-date-object "^1.0.1"
is-symbol "^1.0.2"
+es5-ext@^0.10.35, es5-ext@^0.10.50, es5-ext@~0.10.14:
+ version "0.10.62"
+ resolved "https://registry.yarnpkg.com/es5-ext/-/es5-ext-0.10.62.tgz#5e6adc19a6da524bf3d1e02bbc8960e5eb49a9a5"
+ integrity sha512-BHLqn0klhEpnOKSrzn/Xsz2UIW8j+cGmo9JLzr8BiUapV8hPL9+FliFqjwr9ngW7jWdnxv6eO+/LqyhJVqgrjA==
+ dependencies:
+ es6-iterator "^2.0.3"
+ es6-symbol "^3.1.3"
+ next-tick "^1.1.0"
+
es5-shim@^4.5.13:
version "4.5.15"
resolved "https://registry.yarnpkg.com/es5-shim/-/es5-shim-4.5.15.tgz#6a26869b261854a3b045273f5583c52d390217fe"
integrity sha512-FYpuxEjMeDvU4rulKqFdukQyZSTpzhg4ScQHrAosrlVpR6GFyaw14f74yn2+4BugniIS0Frpg7TvwZocU4ZMTw==
+es6-iterator@^2.0.3:
+ version "2.0.3"
+ resolved "https://registry.yarnpkg.com/es6-iterator/-/es6-iterator-2.0.3.tgz#a7de889141a05a94b0854403b2d0a0fbfa98f3b7"
+ integrity sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==
+ dependencies:
+ d "1"
+ es5-ext "^0.10.35"
+ es6-symbol "^3.1.1"
+
es6-promise@^4.0.3:
version "4.2.8"
resolved "https://registry.yarnpkg.com/es6-promise/-/es6-promise-4.2.8.tgz#4eb21594c972bc40553d276e510539143db53e0a"
@@ -10006,6 +10064,14 @@ es6-shim@^0.35.5:
resolved "https://registry.yarnpkg.com/es6-shim/-/es6-shim-0.35.6.tgz#d10578301a83af2de58b9eadb7c2c9945f7388a0"
integrity sha512-EmTr31wppcaIAgblChZiuN/l9Y7DPyw8Xtbg7fIVngn6zMW+IEBJDJngeKC3x6wr0V/vcA2wqeFnaw1bFJbDdA==
+es6-symbol@^3.1.1, es6-symbol@^3.1.3:
+ version "3.1.3"
+ resolved "https://registry.yarnpkg.com/es6-symbol/-/es6-symbol-3.1.3.tgz#bad5d3c1bcdac28269f4cb331e431c78ac705d18"
+ integrity sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==
+ dependencies:
+ d "^1.0.1"
+ ext "^1.1.2"
+
escalade@^3.0.2, escalade@^3.1.1:
version "3.1.1"
resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.1.tgz#d8cfdc7000965c5a0174b4a82eaa5c0552742e40"
@@ -10344,6 +10410,14 @@ etag@^1.8.1, etag@~1.8.1:
resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
integrity sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=
+event-emitter@^0.3.5:
+ version "0.3.5"
+ resolved "https://registry.yarnpkg.com/event-emitter/-/event-emitter-0.3.5.tgz#df8c69eef1647923c7157b9ce83840610b02cc39"
+ integrity sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==
+ dependencies:
+ d "1"
+ es5-ext "~0.10.14"
+
eventemitter3@^4.0.0, eventemitter3@^4.0.7:
version "4.0.7"
resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.7.tgz#2de9b68f6528d5644ef5c59526a1b4a07306169f"
@@ -10488,6 +10562,13 @@ express@^4.17.1:
utils-merge "1.0.1"
vary "~1.1.2"
+ext@^1.1.2:
+ version "1.7.0"
+ resolved "https://registry.yarnpkg.com/ext/-/ext-1.7.0.tgz#0ea4383c0103d60e70be99e9a7f11027a33c4f5f"
+ integrity sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==
+ dependencies:
+ type "^2.7.2"
+
extend-shallow@^2.0.1:
version "2.0.1"
resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-2.0.1.tgz#51af7d614ad9a9f610ea1bafbb989d6b1c56890f"
@@ -13693,6 +13774,16 @@ jstransformer@1.0.0:
is-promise "^2.0.0"
promise "^7.0.1"
+jszip@^3.7.1:
+ version "3.10.1"
+ resolved "https://registry.yarnpkg.com/jszip/-/jszip-3.10.1.tgz#34aee70eb18ea1faec2f589208a157d1feb091c2"
+ integrity sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g==
+ dependencies:
+ lie "~3.3.0"
+ pako "~1.0.2"
+ readable-stream "~2.3.6"
+ setimmediate "^1.0.5"
+
junk@^3.1.0:
version "3.1.0"
resolved "https://registry.yarnpkg.com/junk/-/junk-3.1.0.tgz#31499098d902b7e98c5d9b9c80f43457a88abfa1"
@@ -13941,6 +14032,13 @@ lie@3.1.1:
dependencies:
immediate "~3.0.5"
+lie@~3.3.0:
+ version "3.3.0"
+ resolved "https://registry.yarnpkg.com/lie/-/lie-3.3.0.tgz#dcf82dee545f46074daf200c7c1c5a08e0f40f6a"
+ integrity sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==
+ dependencies:
+ immediate "~3.0.5"
+
lilconfig@^2.0.3:
version "2.0.3"
resolved "https://registry.yarnpkg.com/lilconfig/-/lilconfig-2.0.3.tgz#68f3005e921dafbd2a2afb48379986aa6d2579fd"
@@ -14014,7 +14112,7 @@ loader-utils@^1.0.2, loader-utils@^1.1.0, loader-utils@^1.2.3, loader-utils@^1.4
emojis-list "^3.0.0"
json5 "^1.0.1"
-localforage@^1.8.1:
+localforage@*, localforage@^1.10.0, localforage@^1.8.1:
version "1.10.0"
resolved "https://registry.yarnpkg.com/localforage/-/localforage-1.10.0.tgz#5c465dc5f62b2807c3a84c0c6a1b1b3212781dd4"
integrity sha512-14/H1aX7hzBBmmh7sGPd+AOMkkIrHM3Z1PAyGgZigA1H1p5O5ANnMyWzvpAETtG68/dC4pC0ncy3+PPGzXZHPg==
@@ -14321,6 +14419,11 @@ markdown-to-jsx@^7.1.3:
resolved "https://registry.yarnpkg.com/markdown-to-jsx/-/markdown-to-jsx-7.1.3.tgz#f00bae66c0abe7dd2d274123f84cb6bd2a2c7c6a"
integrity sha512-jtQ6VyT7rMT5tPV0g2EJakEnXLiPksnvlYtwQsVVZ611JsWGN8bQ1tVSDX4s6JllfEH6wmsYxNjTUAMrPmNA8w==
+marks-pane@^1.0.9:
+ version "1.0.9"
+ resolved "https://registry.yarnpkg.com/marks-pane/-/marks-pane-1.0.9.tgz#c0b5ab813384d8cd81faaeb3bbf3397dc809c1b3"
+ integrity sha512-Ahs4oeG90tbdPWwAJkAAoHg2lRR8lAs9mZXETNPO9hYg3AkjUJBKi1NQ4aaIQZVGrig7c/3NUV1jANl8rFTeMg==
+
mathml-tag-names@^2.1.3:
version "2.1.3"
resolved "https://registry.yarnpkg.com/mathml-tag-names/-/mathml-tag-names-2.1.3.tgz#4ddadd67308e780cf16a47685878ee27b736a0a3"
@@ -14986,6 +15089,11 @@ nested-error-stacks@^2.0.0, nested-error-stacks@^2.1.0:
resolved "https://registry.yarnpkg.com/nested-error-stacks/-/nested-error-stacks-2.1.0.tgz#0fbdcf3e13fe4994781280524f8b96b0cdff9c61"
integrity sha512-AO81vsIO1k1sM4Zrd6Hu7regmJN1NSiAja10gc4bX3F0wd+9rQmcuHQaHVQCYIEC8iFXnE+mavh23GOt7wBgug==
+next-tick@^1.1.0:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/next-tick/-/next-tick-1.1.0.tgz#1836ee30ad56d67ef281b22bd199f709449b35eb"
+ integrity sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==
+
nice-try@^1.0.4:
version "1.0.5"
resolved "https://registry.yarnpkg.com/nice-try/-/nice-try-1.0.5.tgz#a3378a7696ce7d223e88fc9b764bd7ef1089e366"
@@ -15638,7 +15746,7 @@ p5@^1.4.0:
resolved "https://registry.yarnpkg.com/p5/-/p5-1.4.0.tgz#d4d0f001c297525831861af5e017cdb5ef517a75"
integrity sha512-U888W2ChcIzPhRhnv4FkNhaa4f5BDIWZfLhzvx9ZrQ5KtkZr/+o1UPIicV3yWTRy0HEG23NviHyDR3kgjaJ9wA==
-pako@1.0.11, pako@~1.0.5:
+pako@1.0.11, pako@~1.0.2, pako@~1.0.5:
version "1.0.11"
resolved "https://registry.yarnpkg.com/pako/-/pako-1.0.11.tgz#6c9599d340d54dfd3946380252a35705a6b992bf"
integrity sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==
@@ -15850,6 +15958,11 @@ path-type@^4.0.0:
resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b"
integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==
+path-webpack@0.0.3:
+ version "0.0.3"
+ resolved "https://registry.yarnpkg.com/path-webpack/-/path-webpack-0.0.3.tgz#ff6dec749eec5a94605c04d5f63fc55607a03a16"
+ integrity sha512-AmeDxedoo5svf7aB3FYqSAKqMxys014lVKBzy1o/5vv9CtU7U4wgGWL1dA2o6MOzcD53ScN4Jmiq6VbtLz1vIQ==
+
pbkdf2@^3.0.16, pbkdf2@^3.0.3, pbkdf2@^3.0.9, pbkdf2@^3.1.1:
version "3.1.2"
resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.1.2.tgz#dd822aa0887580e52f1a039dc3eda108efae3075"
@@ -18601,7 +18714,7 @@ set-value@^2.0.0, set-value@^2.0.1:
is-plain-object "^2.0.3"
split-string "^3.0.1"
-setimmediate@^1.0.4:
+setimmediate@^1.0.4, setimmediate@^1.0.5:
version "1.0.5"
resolved "https://registry.yarnpkg.com/setimmediate/-/setimmediate-1.0.5.tgz#290cbb232e306942d7d7ea9b83732ab7856f8285"
integrity sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=
@@ -20086,6 +20199,16 @@ type-tagger@^1.0.0:
resolved "https://registry.yarnpkg.com/type-tagger/-/type-tagger-1.0.0.tgz#dc6297e52e17097c1b92b42c16816a18f631e7f4"
integrity sha512-FIPqqpmDgdaulCnRoKv1/d3U4xVBUrYn42QXWNP3XYmgfPUDuBUsgFOb9ntT0aIe0UsUP+lknpQ5d9Kn36RssA==
+type@^1.0.1:
+ version "1.2.0"
+ resolved "https://registry.yarnpkg.com/type/-/type-1.2.0.tgz#848dd7698dafa3e54a6c479e759c4bc3f18847a0"
+ integrity sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==
+
+type@^2.7.2:
+ version "2.7.2"
+ resolved "https://registry.yarnpkg.com/type/-/type-2.7.2.tgz#2376a15a3a28b1efa0f5350dcf72d24df6ef98d0"
+ integrity sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==
+
typedarray-to-buffer@3.1.5, typedarray-to-buffer@^3.1.5:
version "3.1.5"
resolved "https://registry.yarnpkg.com/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz#a97ee7a9ff42691b9f783ff1bc5112fe3fca9080"