357 lines
9.2 KiB
TypeScript
357 lines
9.2 KiB
TypeScript
"use strict";
|
|
import { isNil, map } from "lodash";
|
|
import {
|
|
Context,
|
|
Service,
|
|
ServiceBroker,
|
|
ServiceSchema,
|
|
Errors,
|
|
} from "moleculer";
|
|
import { DbMixin } from "../mixins/db.mixin";
|
|
import Comic from "../models/comic.model";
|
|
import { walkFolder } from "../utils/file.utils";
|
|
import { convertXMLToJSON } from "../utils/xml.utils";
|
|
import https from "https";
|
|
import { logger } from "../utils/logger.utils";
|
|
import { sendToRabbitMQ } from "../queue/importQueue";
|
|
import {
|
|
IExtractComicBookCoverErrorResponse,
|
|
IExtractedComicBookCoverFile,
|
|
} from "threetwo-ui-typings";
|
|
import { extractCoverFromFile } from "../utils/uncompression.utils";
|
|
const ObjectId = require("mongoose").Types.ObjectId;
|
|
|
|
export default class ImportService extends Service {
|
|
public constructor(
|
|
public broker: ServiceBroker,
|
|
schema: ServiceSchema<{}> = { name: "import" }
|
|
) {
|
|
super(broker);
|
|
this.parseServiceSchema(
|
|
Service.mergeSchemas(
|
|
{
|
|
name: "import",
|
|
mixins: [DbMixin("comics", Comic)],
|
|
settings: {
|
|
// Available fields in the responses
|
|
fields: ["_id", "name", "quantity", "price"],
|
|
|
|
// Validator for the `create` & `insert` actions.
|
|
entityValidator: {
|
|
name: "string|min:3",
|
|
price: "number|positive",
|
|
},
|
|
},
|
|
hooks: {},
|
|
actions: {
|
|
walkFolders: {
|
|
rest: "POST /walkFolders",
|
|
params: {
|
|
basePathToWalk: "string",
|
|
},
|
|
async handler(
|
|
ctx: Context<{ basePathToWalk: string }>
|
|
) {
|
|
return await walkFolder(
|
|
ctx.params.basePathToWalk
|
|
);
|
|
},
|
|
},
|
|
convertXMLToJSON: {
|
|
rest: "POST /convertXmlToJson",
|
|
params: {},
|
|
async handler(ctx: Context<{}>) {
|
|
return convertXMLToJSON("lagos");
|
|
},
|
|
},
|
|
importComicsToDb: {
|
|
rest: "POST /importComicsToDB",
|
|
bulkhead: {
|
|
enabled: true,
|
|
concurrency: 50,
|
|
maxQueueSize: 100,
|
|
},
|
|
params: {},
|
|
async handler(
|
|
ctx: Context<{
|
|
extractionOptions: any;
|
|
walkedFolders: [
|
|
{
|
|
name: string;
|
|
extension: string;
|
|
containedIn: string;
|
|
fileSize: number;
|
|
isFile: boolean;
|
|
isLink: boolean;
|
|
}
|
|
];
|
|
}>
|
|
) {
|
|
try {
|
|
const { extractionOptions, walkedFolders } =
|
|
ctx.params;
|
|
map(walkedFolders, async (folder, idx) => {
|
|
let comicExists = await Comic.exists({
|
|
"rawFileDetails.name": `${folder.name}`,
|
|
});
|
|
if (!comicExists) {
|
|
// 1. Extract cover and cover metadata
|
|
let comicBookCoverMetadata:
|
|
| IExtractedComicBookCoverFile
|
|
| IExtractComicBookCoverErrorResponse
|
|
| IExtractedComicBookCoverFile[] =
|
|
await extractCoverFromFile(
|
|
extractionOptions,
|
|
folder
|
|
);
|
|
|
|
// 2. Add to mongo
|
|
const dbImportResult =
|
|
await this.broker.call(
|
|
"import.rawImportToDB",
|
|
{
|
|
importStatus: {
|
|
isImported: true,
|
|
tagged: false,
|
|
matchedResult: {
|
|
score: "0",
|
|
},
|
|
},
|
|
rawFileDetails:
|
|
comicBookCoverMetadata,
|
|
sourcedMetadata: {
|
|
comicvine: {},
|
|
},
|
|
},
|
|
{}
|
|
);
|
|
// 3. Send to the queue
|
|
sendToRabbitMQ(
|
|
"comicBookCovers",
|
|
JSON.stringify({
|
|
comicBookCoverMetadata,
|
|
dbImportResult,
|
|
})
|
|
);
|
|
} else {
|
|
logger.info(
|
|
`Comic: \"${folder.name}\" already exists in the database`
|
|
);
|
|
}
|
|
});
|
|
} catch (error) {
|
|
logger.error(
|
|
"Error importing comic books",
|
|
error
|
|
);
|
|
}
|
|
},
|
|
},
|
|
rawImportToDB: {
|
|
rest: "POST /rawImportToDB",
|
|
params: {},
|
|
async handler(
|
|
ctx: Context<{
|
|
sourcedMetadata: {
|
|
comicvine: {
|
|
volume: { api_detail_url: string };
|
|
volumeInformation: {};
|
|
};
|
|
};
|
|
rawFileDetails: {
|
|
name: string;
|
|
};
|
|
}>
|
|
) {
|
|
let volumeDetails;
|
|
const comicMetadata = ctx.params;
|
|
if (
|
|
comicMetadata.sourcedMetadata.comicvine &&
|
|
!isNil(
|
|
comicMetadata.sourcedMetadata.comicvine
|
|
.volume
|
|
)
|
|
) {
|
|
volumeDetails =
|
|
await this.getComicVineVolumeMetadata(
|
|
comicMetadata.sourcedMetadata
|
|
.comicvine.volume.api_detail_url
|
|
);
|
|
comicMetadata.sourcedMetadata.comicvine.volumeInformation =
|
|
volumeDetails;
|
|
}
|
|
return new Promise(async (resolve, reject) => {
|
|
Comic.create(ctx.params, (error, data) => {
|
|
if (data) {
|
|
resolve(data);
|
|
} else if (error) {
|
|
throw new Errors.MoleculerError(
|
|
"Failed to import comic book",
|
|
400,
|
|
"IMS_FAILED_COMIC_BOOK_IMPORT",
|
|
data
|
|
);
|
|
}
|
|
});
|
|
});
|
|
},
|
|
},
|
|
applyComicVineMetadata: {
|
|
rest: "POST /applyComicVineMetadata",
|
|
params: {},
|
|
async handler(
|
|
ctx: Context<{
|
|
match: {
|
|
volume: { api_detail_url: string };
|
|
volumeInformation: object;
|
|
};
|
|
comicObjectId: string;
|
|
}>
|
|
) {
|
|
// 1. Find mongo object by id
|
|
// 2. Import payload into sourcedMetadata.comicvine
|
|
const comicObjectId = new ObjectId(
|
|
ctx.params.comicObjectId
|
|
);
|
|
const matchedResult = ctx.params.match;
|
|
let volumeDetailsPromise;
|
|
if (!isNil(matchedResult.volume)) {
|
|
volumeDetailsPromise =
|
|
this.getComicVineVolumeMetadata(
|
|
matchedResult.volume.api_detail_url
|
|
);
|
|
}
|
|
return new Promise(async (resolve, reject) => {
|
|
const volumeDetails =
|
|
await volumeDetailsPromise;
|
|
matchedResult.volumeInformation =
|
|
volumeDetails;
|
|
Comic.findByIdAndUpdate(
|
|
comicObjectId,
|
|
{
|
|
sourcedMetadata: {
|
|
comicvine: matchedResult,
|
|
},
|
|
},
|
|
{ new: true },
|
|
(err, result) => {
|
|
if (err) {
|
|
console.log(err);
|
|
reject(err);
|
|
} else {
|
|
// 3. Fetch and append volume information
|
|
resolve(result);
|
|
}
|
|
}
|
|
);
|
|
});
|
|
},
|
|
},
|
|
applyAirDCPPDownloadMetadata: {
|
|
rest: "POST /applyAirDCPPDownloadMetadata",
|
|
params: {},
|
|
async handler(
|
|
ctx: Context<{
|
|
comicObjectId: string;
|
|
resultId: string;
|
|
bundleId: string;
|
|
directoryIds: [];
|
|
searchInstanceId: string;
|
|
}>
|
|
) {
|
|
const comicObjectId = new ObjectId(
|
|
ctx.params.comicObjectId
|
|
);
|
|
return new Promise((resolve, reject) => {
|
|
Comic.findByIdAndUpdate(
|
|
comicObjectId,
|
|
{
|
|
$push: {
|
|
"acquisition.directconnect": {
|
|
resultId:
|
|
ctx.params.resultId,
|
|
bundleId:
|
|
ctx.params.bundleId,
|
|
directoryIds:
|
|
ctx.params.directoryIds,
|
|
searchInstanceId:
|
|
ctx.params
|
|
.searchInstanceId,
|
|
},
|
|
},
|
|
},
|
|
{ new: true, safe: true, upsert: true },
|
|
(err, result) => {
|
|
if (err) {
|
|
reject(err);
|
|
} else {
|
|
resolve(result);
|
|
}
|
|
}
|
|
);
|
|
});
|
|
},
|
|
},
|
|
|
|
getComicBooks: {
|
|
rest: "POST /getComicBooks",
|
|
params: {},
|
|
async handler(
|
|
ctx: Context<{ paginationOptions: object }>
|
|
) {
|
|
return await Comic.paginate(
|
|
{},
|
|
ctx.params.paginationOptions
|
|
);
|
|
},
|
|
},
|
|
getComicBookById: {
|
|
rest: "POST /getComicBookById",
|
|
params: { id: "string" },
|
|
async handler(ctx: Context<{ id: string }>) {
|
|
return await Comic.findById(ctx.params.id);
|
|
},
|
|
},
|
|
axn: {
|
|
rest: "GET /axn",
|
|
params: {},
|
|
async handler(ctx: Context<{path, stats}>) {
|
|
logger.info(ctx.params);
|
|
return {"pandurang": "hari"};
|
|
}
|
|
},
|
|
},
|
|
methods: {
|
|
getComicVineVolumeMetadata: (apiDetailURL) =>
|
|
new Promise((resolve, reject) =>
|
|
https
|
|
.get(
|
|
`${apiDetailURL}?api_key=a5fa0663683df8145a85d694b5da4b87e1c92c69&format=json&limit=1&offset=0&field_list=id,name,description,image,first_issue,last_issue,publisher,count_of_issues,character_credits,person_credits,aliases`,
|
|
(resp) => {
|
|
let data = "";
|
|
resp.on("data", (chunk) => {
|
|
data += chunk;
|
|
});
|
|
|
|
resp.on("end", () => {
|
|
const volumeInformation =
|
|
JSON.parse(data);
|
|
resolve(
|
|
volumeInformation.results
|
|
);
|
|
});
|
|
}
|
|
)
|
|
.on("error", (err) => {
|
|
console.log("Error: " + err.message);
|
|
reject(err);
|
|
})
|
|
),
|
|
},
|
|
},
|
|
schema
|
|
)
|
|
);
|
|
}
|
|
}
|