mirror of
https://github.com/barthuijgen/factorio-sites.git
synced 2025-02-09 14:33:12 +02:00
Added several ways of generating images, factorioprints scraper and a lot of updates to the main site
This commit is contained in:
parent
e767407b65
commit
1cdaeb7fec
@ -1,2 +1,6 @@
|
||||
node_modules
|
||||
*.Dockerfile
|
||||
*.Dockerfile
|
||||
/credentials
|
||||
/dist
|
||||
/.vscode
|
||||
/.cache
|
4
.gitignore
vendored
4
.gitignore
vendored
@ -37,3 +37,7 @@ testem.log
|
||||
# System Files
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# custom
|
||||
/.cache
|
||||
/credentials
|
6
CREDITS.md
Normal file
6
CREDITS.md
Normal file
@ -0,0 +1,6 @@
|
||||
### Factorio blueprint editor
|
||||
|
||||
https://teoxoy.github.io/factorio-blueprint-editor/
|
||||
LISENCE: MIT
|
||||
|
||||
Used for generating images based on blueprint string
|
@ -12,6 +12,8 @@ Run `nx serve my-app` for a dev server. Navigate to http://localhost:4200/. The
|
||||
|
||||
Run `nx build my-app` to build the project. The build artifacts will be stored in the `dist/` directory. Use the `--prod` flag for a production build.
|
||||
|
||||
### blueprints app
|
||||
|
||||
## Running unit tests
|
||||
|
||||
Run `nx test my-app` to execute the unit tests via [Jest](https://jestjs.io).
|
||||
|
125
apps/blueprint-image-function/src/localFileUpload.ts
Normal file
125
apps/blueprint-image-function/src/localFileUpload.ts
Normal file
@ -0,0 +1,125 @@
|
||||
import * as fs from "fs";
|
||||
import * as path from "path";
|
||||
import { promisify } from "util";
|
||||
import * as sharp from "sharp";
|
||||
import {
|
||||
hasBlueprintImage,
|
||||
getBlueprintByImageHash,
|
||||
saveBlueprintImage,
|
||||
} from "@factorio-sites/database";
|
||||
|
||||
if (!process.env.DIR) throw Error("no 'DIR' environment variable")
|
||||
|
||||
// const fsReaddir = promisify(fs.readdir);
|
||||
const fsReadFile = promisify(fs.readFile);
|
||||
const fsUnlink = promisify(fs.unlink);
|
||||
const FILE_DIR = path.normalize(process.env.DIR);
|
||||
|
||||
const RESIZE_ENABLED = false;
|
||||
|
||||
const uploadFile = async (image_path: string) => {
|
||||
const image_hash = path.basename(image_path, ".png");
|
||||
|
||||
if (!(await getBlueprintByImageHash(image_hash))) {
|
||||
console.log(`Image ${image_hash} has no database record, skipping...`);
|
||||
return;
|
||||
}
|
||||
if (image_hash.includes("(")) {
|
||||
console.log(`Image ${image_hash} is a duplicate, deleting...`);
|
||||
await fsUnlink(image_path);
|
||||
return;
|
||||
}
|
||||
if (await hasBlueprintImage(image_hash)) {
|
||||
console.log(`Image ${image_hash} already exists, deleting...`);
|
||||
await fsUnlink(image_path);
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(`Processing ${image_hash}...`);
|
||||
const image = await fsReadFile(image_path);
|
||||
|
||||
// const calculateImageSizeMod = (pixels: number) =>
|
||||
// Math.min(Math.max((-pixels + 500) / 20500 + 1, 0.3), 1);
|
||||
|
||||
const calculateImageSizeMod = (pixels: number) =>
|
||||
Math.min(Math.max((-pixels + 3000) / 33000 + 1, 0.3), 1);
|
||||
|
||||
let sharp_image = sharp(image);
|
||||
if (RESIZE_ENABLED) {
|
||||
const MAX_IMAGE_DIMENTION = 5000;
|
||||
sharp_image = await sharp_image
|
||||
.metadata()
|
||||
.then((meta) => {
|
||||
if (
|
||||
meta.width &&
|
||||
meta.height &&
|
||||
(meta.width > MAX_IMAGE_DIMENTION || meta.height > MAX_IMAGE_DIMENTION)
|
||||
) {
|
||||
const mod = calculateImageSizeMod(Math.max(meta.width, meta.height));
|
||||
console.log({
|
||||
file: image_path,
|
||||
width: meta.width,
|
||||
height: meta.height,
|
||||
mod,
|
||||
size_mb: image.byteLength / 1024_000,
|
||||
});
|
||||
return sharp_image.resize({
|
||||
width: Math.round(meta.width * mod),
|
||||
height: Math.round(meta.height * mod),
|
||||
});
|
||||
}
|
||||
return sharp_image;
|
||||
})
|
||||
.then((image) => image.webp({ lossless: true }));
|
||||
} else {
|
||||
sharp_image = sharp_image.webp({ lossless: true });
|
||||
}
|
||||
|
||||
const min_image = await sharp_image.toBuffer();
|
||||
|
||||
console.log({
|
||||
input_size_mb: image.byteLength / 1024_000,
|
||||
output_size_mb: min_image.byteLength / 1024_000,
|
||||
});
|
||||
|
||||
// console.log(`Image ${image_hash} processed, writing...`);
|
||||
// fs.writeFileSync(`${image_path}.webp`, min_image);
|
||||
console.log(`Image ${image_hash} processed, uploading...`);
|
||||
await saveBlueprintImage(image_hash, min_image);
|
||||
await fsUnlink(image_path);
|
||||
};
|
||||
|
||||
export async function uploadLocalFiles() {
|
||||
// console.log(`Reading directory`, FILE_DIR);
|
||||
// const files = await fsReaddir(FILE_DIR);
|
||||
// for (let i = 0; i < files.length; i++) {
|
||||
// if (fs.statSync(path.join(FILE_DIR, files[i])).isDirectory()) continue;
|
||||
// await uploadFile(path.join(FILE_DIR, files[i]));
|
||||
// }
|
||||
console.log(`Watching directory`, FILE_DIR);
|
||||
const work_done_buffeer: string[] = [];
|
||||
const work_buffer: string[] = [];
|
||||
fs.watch(FILE_DIR, (type, file) => {
|
||||
if (type === "change" && file && file.endsWith(".png")) {
|
||||
const file_path = path.join(FILE_DIR, file);
|
||||
if (work_buffer.includes(file_path) || work_done_buffeer.includes(file_path)) {
|
||||
return;
|
||||
}
|
||||
work_buffer.push(file_path);
|
||||
}
|
||||
});
|
||||
|
||||
let working = false;
|
||||
const doWork = async () => {
|
||||
if (working || !work_buffer.length) return setTimeout(doWork, 1000);
|
||||
working = true;
|
||||
const file_path = work_buffer.shift();
|
||||
if (file_path) {
|
||||
await uploadFile(file_path);
|
||||
work_done_buffeer.push(file_path);
|
||||
}
|
||||
working = false;
|
||||
doWork();
|
||||
};
|
||||
doWork();
|
||||
}
|
@ -1,15 +1,142 @@
|
||||
import { generateScreenshot } from "@factorio-sites/generate-bp-image";
|
||||
import {
|
||||
saveBlueprintImage,
|
||||
hasBlueprintImage,
|
||||
getBlueprintById,
|
||||
BlueprintEntry,
|
||||
getBlueprintImageRequestTopic,
|
||||
PubSubMessage,
|
||||
getPaginatedBlueprints,
|
||||
} from "@factorio-sites/database";
|
||||
import { environment } from "./environments/environment";
|
||||
import { uploadLocalFiles } from "./localFileUpload";
|
||||
|
||||
const generateImageForSource = async (blueprint: BlueprintEntry) => {
|
||||
if (await hasBlueprintImage(blueprint.image_hash)) {
|
||||
throw Error("Image already exists");
|
||||
}
|
||||
|
||||
const imageBuffer = await generateScreenshot(
|
||||
blueprint,
|
||||
environment.production ? "/tmp" : undefined
|
||||
);
|
||||
|
||||
if (!imageBuffer) return false;
|
||||
|
||||
await saveBlueprintImage(blueprint.image_hash, imageBuffer);
|
||||
|
||||
console.log(`[generateImageForSource] image hash ${blueprint.image_hash} successfully saved`);
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
type Handler = (req: any, res: any) => void; // Don't want to install express types just for this
|
||||
|
||||
/**
|
||||
* Handler method supports Google cloud functions
|
||||
* @param req express request object
|
||||
* @param res express response object
|
||||
*/
|
||||
export const handler: Handler = async (req, res) => {
|
||||
if (!req.query.source) {
|
||||
return res.status(400).end("No source string given");
|
||||
}
|
||||
const string = (req.query.source as string).replace(/ /g, "+");
|
||||
const stream = await generateScreenshot(string, "/tmp");
|
||||
res.status(200);
|
||||
res.setHeader("content-type", "image/png");
|
||||
stream.pipe(res);
|
||||
|
||||
// generateImageForSource((req.query.source as string).replace(/ /g, "+"))
|
||||
// .then(() => {
|
||||
// res.status(200).end("done");
|
||||
// })
|
||||
// .catch((reason) => {
|
||||
// res.status(200).end(reason.message);
|
||||
// });
|
||||
};
|
||||
|
||||
async function subscribeToPubSub() {
|
||||
// Wait to make sure puppeteer browser started
|
||||
await new Promise((resolve) => setTimeout(resolve, 2000));
|
||||
|
||||
const topic = getBlueprintImageRequestTopic();
|
||||
const [subscription] = await topic
|
||||
.subscription("blueprint-image-function-app", {
|
||||
flowControl: { allowExcessMessages: false, maxMessages: 1, maxExtension: 3600 },
|
||||
})
|
||||
.get();
|
||||
|
||||
console.log(`[pubsub] Listening to subscription`);
|
||||
|
||||
let handlerBusy = false;
|
||||
const messageHandler = async (message: PubSubMessage) => {
|
||||
if (!handlerBusy) handlerBusy = true;
|
||||
else {
|
||||
console.log(`nack'd message because handler is busy ${message.data.toString()}`);
|
||||
return message.nack();
|
||||
}
|
||||
|
||||
try {
|
||||
const data = JSON.parse(message.data.toString());
|
||||
if (!data.blueprintId) return console.log("blueprintId not found in message body");
|
||||
console.log("------------------------------------------------");
|
||||
console.log("[pubsub] generating image for", data.blueprintId);
|
||||
const blueprint = await getBlueprintById(data.blueprintId);
|
||||
if (!blueprint) return console.log("Blueprint not found");
|
||||
const start_time = Date.now();
|
||||
await generateImageForSource(blueprint);
|
||||
const duration = Date.now() - start_time;
|
||||
console.log(`[pubsub] image generated in ${duration}ms`);
|
||||
message.ack();
|
||||
|
||||
if (duration > 30000) {
|
||||
console.log("Process too slow, closing...");
|
||||
subscription.off("message", messageHandler);
|
||||
return setTimeout(() => process.exit(1), 1000);
|
||||
}
|
||||
} catch (reason) {
|
||||
if (reason.message === "Image already exists") {
|
||||
console.log(`Image already exists`);
|
||||
message.ack();
|
||||
} else if (reason.message === "Failed to parse blueprint string") {
|
||||
console.log(`Blueprint editor could not handle string`);
|
||||
message.ack();
|
||||
} else {
|
||||
console.error("[pubsub:error]", reason);
|
||||
message.nack();
|
||||
}
|
||||
}
|
||||
handlerBusy = false;
|
||||
};
|
||||
|
||||
subscription.on("message", messageHandler);
|
||||
// image hash = a99525f97c26c7242ecdd96679043b1a5e65dd0c
|
||||
// SELECT * FROM BlueprintBook WHERE blueprint_ids CONTAINS Key(Blueprint, 4532736400293888)
|
||||
// bp = Key(Blueprint, 4532736400293888)
|
||||
// book = Key(BlueprintBook, 5034207050989568)
|
||||
// page = 6225886932107264
|
||||
|
||||
subscription.on("error", (error) => {
|
||||
console.error("[pubsub] Received error:", error);
|
||||
});
|
||||
}
|
||||
|
||||
async function rePublishAllBlueprints() {
|
||||
const topic = getBlueprintImageRequestTopic();
|
||||
const fetchPage = async (page = 1) => {
|
||||
const blueprints = await getPaginatedBlueprints(page);
|
||||
if (blueprints.length === 0) {
|
||||
return console.log("No more blueprints found");
|
||||
}
|
||||
console.log(`Publishing page ${page} with ${blueprints.length} blueprints`);
|
||||
|
||||
await Promise.all(
|
||||
blueprints.map((blueprint) => {
|
||||
return topic.publishJSON({ blueprintId: blueprint.id });
|
||||
})
|
||||
);
|
||||
fetchPage(page + 1);
|
||||
};
|
||||
await fetchPage();
|
||||
}
|
||||
|
||||
uploadLocalFiles().catch((reason) => console.error("Fatal error:", reason));
|
||||
// subscribeToPubSub().catch((reason) => console.error("Fatal error:", reason));
|
||||
// rePublishAllBlueprints().catch((reason) => console.error("Fatal error:", reason));
|
||||
|
@ -194,10 +194,7 @@
|
||||
"jsx-a11y/accessible-emoji": "warn",
|
||||
"jsx-a11y/alt-text": "warn",
|
||||
"jsx-a11y/anchor-has-content": "warn",
|
||||
"jsx-a11y/anchor-is-valid": [
|
||||
"warn",
|
||||
{ "aspects": ["noHref", "invalidHref"] }
|
||||
],
|
||||
"jsx-a11y/anchor-is-valid": ["warn", { "aspects": ["noHref", "invalidHref"] }],
|
||||
"jsx-a11y/aria-activedescendant-has-tabindex": "warn",
|
||||
"jsx-a11y/aria-props": "warn",
|
||||
"jsx-a11y/aria-proptypes": "warn",
|
||||
@ -231,10 +228,7 @@
|
||||
}
|
||||
],
|
||||
"no-unused-vars": "off",
|
||||
"@typescript-eslint/no-unused-vars": [
|
||||
"warn",
|
||||
{ "args": "none", "ignoreRestSiblings": true }
|
||||
],
|
||||
"@typescript-eslint/no-unused-vars": ["warn", { "args": "none", "ignoreRestSiblings": true }],
|
||||
"no-useless-constructor": "off",
|
||||
"@typescript-eslint/no-useless-constructor": "warn",
|
||||
"@typescript-eslint/no-unused-expressions": [
|
||||
|
16
apps/blueprints/README.md
Normal file
16
apps/blueprints/README.md
Normal file
@ -0,0 +1,16 @@
|
||||
## Before deploying
|
||||
|
||||
Make sure `prod.package.json` is up to date on packages from the root `package.json`
|
||||
|
||||
## Deploying
|
||||
|
||||
`docker build -t eu.gcr.io/factorio-sites/blueprints --file blueprints.Dockerfile .`
|
||||
`docker push eu.gcr.io/factorio-sites/blueprints`
|
||||
|
||||
### Testing deployment locally
|
||||
|
||||
`docker run --rm -p 3000:3000 eu.gcr.io/factorio-sites/blueprints`
|
||||
|
||||
### windows env
|
||||
|
||||
`$env:GOOGLE_APPLICATION_CREDENTIALS="D:\git\factorio-sites\credentials\factorio-sites.json"`
|
6
apps/blueprints/index.d.ts
vendored
6
apps/blueprints/index.d.ts
vendored
@ -1,6 +1,10 @@
|
||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||
declare module '*.svg' {
|
||||
declare module "*.svg" {
|
||||
const content: any;
|
||||
export const ReactComponent: any;
|
||||
export default content;
|
||||
}
|
||||
|
||||
declare module "react-map-interaction" {
|
||||
export const MapInteractionCSS: any;
|
||||
}
|
||||
|
331
apps/blueprints/pages/[blueprint]/[blueprintId].tsx
Normal file
331
apps/blueprints/pages/[blueprint]/[blueprintId].tsx
Normal file
@ -0,0 +1,331 @@
|
||||
/** @jsx jsx */
|
||||
import React, { ReactNode, useEffect, useState } from "react";
|
||||
import { jsx, css } from "@emotion/core";
|
||||
import { NextPage, NextPageContext } from "next";
|
||||
import BBCode from "bbcode-to-react";
|
||||
import { Button, Grid } from "@chakra-ui/core";
|
||||
import {
|
||||
BlueprintBookEntry,
|
||||
BlueprintEntry,
|
||||
BlueprintPageEntry,
|
||||
getBlueprintBookById,
|
||||
getBlueprintById,
|
||||
getBlueprintPageById,
|
||||
hasBlueprintImage,
|
||||
} from "@factorio-sites/database";
|
||||
import { BlueprintData, timeLogger } from "@factorio-sites/common-utils";
|
||||
import { chakraResponsive, parseBlueprintStringClient } from "@factorio-sites/web-utils";
|
||||
import { Panel } from "../../src/Panel";
|
||||
import { Markdown } from "../../src/Markdown";
|
||||
import { FullscreenImage } from "../../src/FullscreenImage";
|
||||
import { BookChildTree } from "../../src/BookChildTree";
|
||||
import { CopyButton } from "../../src/CopyButton";
|
||||
|
||||
const imageStyle = css`
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
&:hover {
|
||||
cursor: pointer;
|
||||
}
|
||||
`;
|
||||
|
||||
type Selected =
|
||||
| { type: "blueprint"; data: Pick<BlueprintEntry, "id" | "blueprint_hash" | "image_hash"> }
|
||||
| { type: "blueprint_book"; data: Pick<BlueprintBookEntry, "id" | "blueprint_hash"> };
|
||||
|
||||
interface IndexProps {
|
||||
image_exists: boolean;
|
||||
selected: Selected;
|
||||
blueprint: BlueprintEntry | null;
|
||||
blueprint_book: BlueprintBookEntry | null;
|
||||
blueprint_page: BlueprintPageEntry;
|
||||
}
|
||||
|
||||
export const Index: NextPage<IndexProps> = ({
|
||||
image_exists,
|
||||
selected,
|
||||
blueprint,
|
||||
blueprint_book,
|
||||
blueprint_page,
|
||||
}) => {
|
||||
const [imageZoom, setImageZoom] = useState(false);
|
||||
const [blueprintString, setBlueprintString] = useState<string | null>(null);
|
||||
const [data, setData] = useState<BlueprintData | null>(null);
|
||||
const [showJson, setShowJson] = useState(false);
|
||||
|
||||
const selectedHash = selected.data.blueprint_hash;
|
||||
|
||||
useEffect(() => {
|
||||
fetch(`/api/string/${selectedHash}`)
|
||||
.then((res) => res.text())
|
||||
.then((string) => {
|
||||
setShowJson(false);
|
||||
setBlueprintString(string);
|
||||
if (selected.type === "blueprint") {
|
||||
const { data } = parseBlueprintStringClient(string);
|
||||
setData(data);
|
||||
} else {
|
||||
setData(null);
|
||||
}
|
||||
});
|
||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||
}, [selectedHash]);
|
||||
|
||||
useEffect(() => {
|
||||
console.log({
|
||||
image_exists,
|
||||
selected,
|
||||
blueprint,
|
||||
blueprint_book,
|
||||
blueprint_page,
|
||||
});
|
||||
}, []);
|
||||
|
||||
const renderImage = () => {
|
||||
let render: ReactNode;
|
||||
if (selected.type === "blueprint_book") {
|
||||
render = <div>Can't show image for a book, select a blueprint to the the image</div>;
|
||||
} else if (!image_exists) {
|
||||
render = <div>The image is not generated yet</div>;
|
||||
} else if (imageZoom) {
|
||||
render = (
|
||||
<FullscreenImage
|
||||
close={() => setImageZoom(false)}
|
||||
alt="blueprint"
|
||||
src={`https://storage.googleapis.com/blueprint-images/${selected.data.image_hash}.webp`}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
render = (
|
||||
<div onClick={() => setImageZoom(true)}>
|
||||
<img
|
||||
alt="blueprint"
|
||||
src={`https://storage.googleapis.com/blueprint-images/${selected.data.image_hash}.webp`}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
return <div css={imageStyle}>{render}</div>;
|
||||
};
|
||||
|
||||
return (
|
||||
<Grid
|
||||
margin="0.7rem"
|
||||
templateColumns={chakraResponsive({ mobile: "1fr", desktop: "1fr 1fr" })}
|
||||
gap={6}
|
||||
>
|
||||
<Panel title={blueprint_page.title} gridColumn="1">
|
||||
{blueprint_book ? (
|
||||
<>
|
||||
<div>This string contains a blueprint book </div>
|
||||
<br />
|
||||
<BookChildTree
|
||||
child_tree={[
|
||||
{
|
||||
id: blueprint_book.id,
|
||||
name: blueprint_book.label,
|
||||
type: "blueprint_book",
|
||||
children: blueprint_book.child_tree,
|
||||
},
|
||||
]}
|
||||
base_url={`/blueprint/${blueprint_page.id}`}
|
||||
selected_id={selected.data.id}
|
||||
/>
|
||||
</>
|
||||
) : blueprint ? (
|
||||
<>
|
||||
<div>This string contains one blueprint</div>
|
||||
<div>tags: {blueprint.tags.join(", ")}</div>
|
||||
</>
|
||||
) : null}
|
||||
</Panel>
|
||||
<Panel
|
||||
title={image_exists ? undefined : "Image"}
|
||||
gridColumn={chakraResponsive({ mobile: "1", desktop: "2" })}
|
||||
gridRow={chakraResponsive({ mobile: "1", desktop: undefined })}
|
||||
>
|
||||
{renderImage()}
|
||||
</Panel>
|
||||
|
||||
<Panel
|
||||
title="Description"
|
||||
gridColumn={chakraResponsive({ mobile: "1", desktop: "1 / span 2" })}
|
||||
>
|
||||
<Markdown>{blueprint_page.description_markdown}</Markdown>
|
||||
</Panel>
|
||||
{selected.type === "blueprint" && data && (
|
||||
<Panel
|
||||
title={(<span>Entities for {BBCode.toReact(data.blueprint.label)}</span>) as any}
|
||||
gridColumn={chakraResponsive({ mobile: "1", desktop: "1 / span 2" })}
|
||||
>
|
||||
<table>
|
||||
<tbody>
|
||||
{Object.entries(
|
||||
data.blueprint.entities.reduce<Record<string, number>>((entities, entity) => {
|
||||
if (entities[entity.name]) {
|
||||
entities[entity.name]++;
|
||||
} else {
|
||||
entities[entity.name] = 1;
|
||||
}
|
||||
return entities;
|
||||
}, {})
|
||||
)
|
||||
.sort((a, b) => b[1] - a[1])
|
||||
.map(([entry_name, entry]) => (
|
||||
<tr key={entry_name} css={{}}>
|
||||
<td css={{ border: "1px solid #909090" }}>
|
||||
<img
|
||||
alt={entry_name.replace(/-/g, " ")}
|
||||
src={`https://factorioprints.com/icons/${entry_name}.png`}
|
||||
/>
|
||||
</td>
|
||||
<td css={{ padding: "5px 10px", border: "1px solid #909090" }}>{entry_name}</td>
|
||||
<td css={{ padding: "5px 10px", border: "1px solid #909090" }}>{entry}</td>
|
||||
</tr>
|
||||
))}
|
||||
</tbody>
|
||||
</table>
|
||||
</Panel>
|
||||
)}
|
||||
<Panel title="string" gridColumn={chakraResponsive({ mobile: "1", desktop: "1 / span 2" })}>
|
||||
<>
|
||||
{blueprintString && <CopyButton content={blueprintString} marginBottom="0.5rem" />}
|
||||
<textarea
|
||||
value={blueprintString || "Loading..."}
|
||||
readOnly
|
||||
css={{
|
||||
width: "100%",
|
||||
height: "100px",
|
||||
resize: "none",
|
||||
color: "#fff",
|
||||
backgroundColor: "#414040",
|
||||
}}
|
||||
/>
|
||||
</>
|
||||
</Panel>
|
||||
<Panel title="json" gridColumn={chakraResponsive({ mobile: "1", desktop: "1 / span 2" })}>
|
||||
{showJson ? (
|
||||
!data ? (
|
||||
<div>Loading...</div>
|
||||
) : (
|
||||
<>
|
||||
<Button
|
||||
variantColor="green"
|
||||
css={{ position: "absolute", right: "65px" }}
|
||||
onClick={() => {
|
||||
setShowJson(false);
|
||||
if (selected.type === "blueprint_book") {
|
||||
setData(null);
|
||||
}
|
||||
}}
|
||||
>
|
||||
hide
|
||||
</Button>
|
||||
<pre css={{ maxHeight: "500px", overflowY: "scroll" }}>
|
||||
{JSON.stringify(data, null, 2)}
|
||||
</pre>
|
||||
</>
|
||||
)
|
||||
) : (
|
||||
<Button
|
||||
variantColor="green"
|
||||
onClick={() => {
|
||||
setShowJson(true);
|
||||
if (selected.type === "blueprint_book") {
|
||||
fetch(`/api/string/${selectedHash}`)
|
||||
.then((res) => res.text())
|
||||
.then((string) => {
|
||||
const { data } = parseBlueprintStringClient(string);
|
||||
setData(data);
|
||||
});
|
||||
}
|
||||
}}
|
||||
>
|
||||
show
|
||||
</Button>
|
||||
)}
|
||||
</Panel>
|
||||
</Grid>
|
||||
);
|
||||
};
|
||||
|
||||
export async function getServerSideProps(context: NextPageContext) {
|
||||
const throwError = (message: string) => {
|
||||
if (!blueprint_page && context.res) {
|
||||
context.res.statusCode = 404;
|
||||
context.res.end({ error: message });
|
||||
return {};
|
||||
}
|
||||
};
|
||||
|
||||
const tl = timeLogger("getServerSideProps");
|
||||
const selected_id = context.query.selected ? (context.query.selected as string) : null;
|
||||
const type = context.query.type ? (context.query.type as string) : null;
|
||||
const blueprintId = context.query.blueprintId ? (context.query.blueprintId as string) : null;
|
||||
|
||||
if (!blueprintId) return throwError("Blueprint ID not found");
|
||||
|
||||
const blueprint_page = await getBlueprintPageById(blueprintId);
|
||||
tl("getBlueprintPageById");
|
||||
|
||||
if (!blueprint_page) return throwError("Blueprint page not found");
|
||||
|
||||
let blueprint: IndexProps["blueprint"] = null;
|
||||
let blueprint_book: IndexProps["blueprint_book"] = null;
|
||||
let selected!: IndexProps["selected"];
|
||||
let selected_blueprint!: BlueprintEntry | null;
|
||||
let selected_blueprint_book!: BlueprintBookEntry | null;
|
||||
|
||||
if (blueprint_page.blueprint_id) {
|
||||
blueprint = await getBlueprintById(blueprint_page.blueprint_id);
|
||||
selected_blueprint = blueprint;
|
||||
tl("getBlueprintById");
|
||||
// blueprint_string = await getBlueprintStringByHash(blueprint.blueprint_hash);
|
||||
} else if (blueprint_page.blueprint_book_id) {
|
||||
blueprint_book = await getBlueprintBookById(blueprint_page.blueprint_book_id);
|
||||
if (selected_id && type === "book") {
|
||||
selected_blueprint_book = await getBlueprintBookById(selected_id);
|
||||
tl("getBlueprintBookById");
|
||||
} else if (selected_id && type !== "book") {
|
||||
selected_blueprint = await getBlueprintById(selected_id);
|
||||
tl("getBlueprintById");
|
||||
} else if (blueprint_book) {
|
||||
selected_blueprint_book = blueprint_book;
|
||||
}
|
||||
}
|
||||
|
||||
if (selected_blueprint) {
|
||||
selected = {
|
||||
type: "blueprint",
|
||||
data: {
|
||||
id: selected_blueprint.id,
|
||||
blueprint_hash: selected_blueprint.blueprint_hash,
|
||||
image_hash: selected_blueprint.image_hash,
|
||||
},
|
||||
};
|
||||
} else if (selected_blueprint_book) {
|
||||
selected = {
|
||||
type: "blueprint_book",
|
||||
data: {
|
||||
id: selected_blueprint_book.id,
|
||||
blueprint_hash: selected_blueprint_book.blueprint_hash,
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
// selected = {type: 'blueprint', data: {id: blueprint.id}}
|
||||
const image_exists =
|
||||
selected.type === "blueprint" ? await hasBlueprintImage(selected.data.image_hash) : false;
|
||||
|
||||
return {
|
||||
props: {
|
||||
image_exists,
|
||||
blueprint,
|
||||
blueprint_book,
|
||||
selected,
|
||||
blueprint_page,
|
||||
} as IndexProps,
|
||||
};
|
||||
}
|
||||
|
||||
export default Index;
|
@ -1,52 +1,78 @@
|
||||
/** @jsx jsx */
|
||||
import React from "react";
|
||||
import { jsx, css, Global } from "@emotion/core";
|
||||
import { AppProps } from "next/app";
|
||||
import Head from "next/head";
|
||||
import { jsx, css } from "@emotion/core";
|
||||
import { normalize } from "./normalize";
|
||||
import { Global } from "@emotion/core";
|
||||
import Router from "next/router";
|
||||
import { CSSReset, ITheme, theme } from "@chakra-ui/core";
|
||||
import { ThemeProvider } from "emotion-theming";
|
||||
import NProgress from "nprogress";
|
||||
import { Header } from "../src/Header";
|
||||
|
||||
const mainStyles = css`
|
||||
font-family: sans-serif;
|
||||
|
||||
header {
|
||||
const globalStyles = css`
|
||||
html {
|
||||
height: 100%;
|
||||
}
|
||||
body {
|
||||
background-color: #201810;
|
||||
background: #201810;
|
||||
background-image: url(https://cdn.factorio.com/assets/img/web/bg_v4-85.jpg),
|
||||
url();
|
||||
background-size: 2048px 3072px;
|
||||
background-position: center top;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: left;
|
||||
background-color: #143055;
|
||||
color: white;
|
||||
padding: 5px 20px;
|
||||
|
||||
h1 {
|
||||
margin: 10px 0;
|
||||
}
|
||||
}
|
||||
|
||||
main {
|
||||
padding: 0 20px;
|
||||
}
|
||||
|
||||
.sidebar {
|
||||
background: #ccc;
|
||||
flex-direction: column;
|
||||
min-height: 100%;
|
||||
font-family: titillium web, sans-serif;
|
||||
}
|
||||
`;
|
||||
|
||||
if (typeof window !== "undefined") {
|
||||
NProgress.configure({
|
||||
showSpinner: false,
|
||||
speed: 800,
|
||||
trickleSpeed: 150,
|
||||
template:
|
||||
'<div class="bar" role="bar" style="background: #00a1ff;position: fixed;z-index: 1031;top: 0;left: 0;width: 100%;height: 4px;" />',
|
||||
});
|
||||
Router.events.on("routeChangeStart", () => NProgress.start());
|
||||
Router.events.on("routeChangeComplete", () => NProgress.done());
|
||||
Router.events.on("routeChangeError", () => NProgress.done());
|
||||
}
|
||||
|
||||
const config = (theme: ITheme) => ({
|
||||
light: {
|
||||
color: theme.colors.gray[800],
|
||||
bg: theme.colors.gray[300],
|
||||
borderColor: theme.colors.gray[200],
|
||||
placeholderColor: theme.colors.gray[500],
|
||||
},
|
||||
dark: {
|
||||
color: theme.colors.whiteAlpha[900],
|
||||
bg: theme.colors.gray[800],
|
||||
borderColor: theme.colors.whiteAlpha[300],
|
||||
placeholderColor: theme.colors.whiteAlpha[400],
|
||||
},
|
||||
});
|
||||
|
||||
const CustomApp = ({ Component, pageProps }: AppProps) => {
|
||||
return (
|
||||
<>
|
||||
<Global styles={normalize} />
|
||||
<ThemeProvider theme={theme}>
|
||||
<Global styles={globalStyles} />
|
||||
<CSSReset config={config} />
|
||||
<Head>
|
||||
<title>Welcome to blueprints!</title>
|
||||
<link
|
||||
href="https://cdn.factorio.com/assets/fonts/titillium-web.css"
|
||||
rel="stylesheet"
|
||||
></link>
|
||||
</Head>
|
||||
<div css={mainStyles}>
|
||||
<header>
|
||||
<h1>Factorio Blueprints</h1>
|
||||
</header>
|
||||
<div>
|
||||
<Header />
|
||||
<main>
|
||||
<Component {...pageProps} />
|
||||
</main>
|
||||
</div>
|
||||
</>
|
||||
</ThemeProvider>
|
||||
);
|
||||
};
|
||||
|
||||
|
65
apps/blueprints/pages/api/image-to-gen.ts
Normal file
65
apps/blueprints/pages/api/image-to-gen.ts
Normal file
@ -0,0 +1,65 @@
|
||||
import { NextApiHandler } from "next";
|
||||
import {
|
||||
getBlueprintImageRequestTopic,
|
||||
PubSubMessage,
|
||||
getBlueprintById,
|
||||
getBlueprintStringByHash,
|
||||
hasBlueprintImage,
|
||||
BlueprintEntry,
|
||||
} from "@factorio-sites/database";
|
||||
|
||||
const getOneMessage = async (): Promise<BlueprintEntry> => {
|
||||
const topic = getBlueprintImageRequestTopic();
|
||||
const [subscription] = await topic
|
||||
.subscription("blueprint-image-function-app", {
|
||||
flowControl: { allowExcessMessages: false, maxMessages: 1 },
|
||||
})
|
||||
.get();
|
||||
|
||||
return new Promise((resolve) => {
|
||||
let acked_msg = false;
|
||||
const messageHandler = async (message: PubSubMessage) => {
|
||||
console.log("[pubsub]" + message.data.toString());
|
||||
if (acked_msg) return;
|
||||
|
||||
const data = JSON.parse(message.data.toString());
|
||||
const blueprint = await getBlueprintById(data.blueprintId);
|
||||
|
||||
if (await hasBlueprintImage(blueprint.image_hash)) {
|
||||
console.log(`Blueprint ${data.blueprintId} image already exists ${blueprint.image_hash}`);
|
||||
return message.ack();
|
||||
}
|
||||
|
||||
if (acked_msg) return; // check again if it changes during async calls
|
||||
acked_msg = true;
|
||||
message.ack();
|
||||
subscription.off("message", messageHandler);
|
||||
resolve(blueprint);
|
||||
};
|
||||
subscription.on("message", messageHandler);
|
||||
});
|
||||
};
|
||||
|
||||
const handler: NextApiHandler = async (req, res) => {
|
||||
// Allow the url to be used in the blueprint editor
|
||||
if (
|
||||
req.headers.origin === "https://teoxoy.github.io" ||
|
||||
req.headers.origin.startsWith("http://localhost")
|
||||
) {
|
||||
res.setHeader("Access-Control-Allow-Origin", req.headers.origin);
|
||||
}
|
||||
|
||||
const blueprint = await getOneMessage();
|
||||
const string = await getBlueprintStringByHash(blueprint.blueprint_hash);
|
||||
|
||||
res.setHeader("Content-Type", "application/json");
|
||||
res.status(200).end(
|
||||
JSON.stringify({
|
||||
blueprintId: blueprint.id,
|
||||
image_hash: blueprint.image_hash,
|
||||
string: string,
|
||||
})
|
||||
);
|
||||
};
|
||||
|
||||
export default handler;
|
43
apps/blueprints/pages/api/post-image.ts
Normal file
43
apps/blueprints/pages/api/post-image.ts
Normal file
@ -0,0 +1,43 @@
|
||||
import { getBlueprintById, saveBlueprintImage } from "@factorio-sites/database";
|
||||
// import imagemin from "imagemin";
|
||||
// import imageminWebp from "imagemin-webp";
|
||||
import { NextApiHandler } from "next";
|
||||
|
||||
const handler: NextApiHandler = async (req, res) => {
|
||||
// Allow the url to be used in the blueprint editor
|
||||
if (
|
||||
(req.method === "OPTIONS" || req.method === "POST") &&
|
||||
req.headers.origin === "https://teoxoy.github.io"
|
||||
) {
|
||||
res.setHeader("Access-Control-Allow-Origin", "https://teoxoy.github.io");
|
||||
res.setHeader("Access-Control-Allow-Headers", "content-type");
|
||||
}
|
||||
if (req.method === "OPTIONS") return res.status(200).end();
|
||||
else if (req.method !== "POST") {
|
||||
return res.status(400).end("Only accepts POST");
|
||||
}
|
||||
|
||||
if (req.body.blueprintId && req.body.image) {
|
||||
console.log(`store image for blueprint ${req.body.blueprintId}`);
|
||||
const buffer = Buffer.from(req.body.image, "base64");
|
||||
const buffermin = buffer;
|
||||
// const buffermin = await imagemin.buffer(buffer, {
|
||||
// plugins: [imageminWebp({ quality: 50 })],
|
||||
// });
|
||||
|
||||
console.log("minified image buffer length", buffermin.byteLength);
|
||||
|
||||
const blueprint = await getBlueprintById(req.body.blueprintId);
|
||||
await saveBlueprintImage(blueprint.image_hash, buffermin);
|
||||
console.log(`Successfuly saved image ${blueprint.image_hash}`);
|
||||
|
||||
res.setHeader("Content-Type", "text/plain");
|
||||
res.status(200).end("ok");
|
||||
return;
|
||||
}
|
||||
|
||||
res.setHeader("Content-Type", "text/plain");
|
||||
res.status(400).end("no post body");
|
||||
};
|
||||
|
||||
export default handler;
|
20
apps/blueprints/pages/api/string/[hash].ts
Normal file
20
apps/blueprints/pages/api/string/[hash].ts
Normal file
@ -0,0 +1,20 @@
|
||||
import { NextApiHandler } from "next";
|
||||
import { getBlueprintStringByHash } from "@factorio-sites/database";
|
||||
|
||||
const handler: NextApiHandler = async (req, res) => {
|
||||
if (!req.query.hash) {
|
||||
return res.status(400).end("No string hash provided");
|
||||
}
|
||||
|
||||
const blueprintString = await getBlueprintStringByHash(req.query.hash as string);
|
||||
|
||||
// Allow the url to be used in the blueprint editor
|
||||
if (req.headers.origin === "https://teoxoy.github.io") {
|
||||
res.setHeader("Access-Control-Allow-Origin", "https://teoxoy.github.io");
|
||||
}
|
||||
res.setHeader("Cache-Control", "public, max-age=86400");
|
||||
res.setHeader("Content-Type", "text/plain");
|
||||
res.status(200).end(blueprintString);
|
||||
};
|
||||
|
||||
export default handler;
|
@ -1,12 +1,64 @@
|
||||
/** @jsx jsx */
|
||||
/* eslint-disable jsx-a11y/anchor-is-valid */
|
||||
import React from "react";
|
||||
import { NextPage, NextPageContext } from "next";
|
||||
import Link from "next/link";
|
||||
import { jsx, css } from "@emotion/core";
|
||||
import { BlueprintPageEntry, getMostRecentBlueprintPages } from "@factorio-sites/database";
|
||||
import { Panel } from "../src/Panel";
|
||||
import { SimpleGrid } from "@chakra-ui/core";
|
||||
import { Pagination } from "../src/Pagination";
|
||||
|
||||
export const Index = () => {
|
||||
/*
|
||||
* Replace the elements below with your own.
|
||||
*
|
||||
* Note: The corresponding styles are in the ./${fileName}.${style} file.
|
||||
*/
|
||||
return <div>app</div>;
|
||||
const linkStyles = css`
|
||||
width: 100%;
|
||||
margin: 5px 0;
|
||||
a {
|
||||
display: block;
|
||||
padding: 5px;
|
||||
color: #fff;
|
||||
}
|
||||
&:hover {
|
||||
cursor: pointer;
|
||||
background: #ccc;
|
||||
}
|
||||
`;
|
||||
|
||||
const BlueprintComponent: React.FC<{ blueprint: BlueprintPageEntry }> = ({ blueprint }) => (
|
||||
<div css={linkStyles}>
|
||||
<Link href={`/blueprint/${blueprint.id}`} passHref>
|
||||
<a>{blueprint.title}</a>
|
||||
</Link>
|
||||
</div>
|
||||
);
|
||||
|
||||
interface IndexProps {
|
||||
page: number;
|
||||
blueprints: BlueprintPageEntry[];
|
||||
}
|
||||
|
||||
export const Index: NextPage<IndexProps> = ({ page, blueprints }) => {
|
||||
return (
|
||||
<SimpleGrid columns={1} margin="0.7rem">
|
||||
<Panel title="Blueprints" w="100%">
|
||||
{blueprints.map((bp) => (
|
||||
<BlueprintComponent key={bp.id} blueprint={bp} />
|
||||
))}
|
||||
<Pagination page={page} />
|
||||
</Panel>
|
||||
</SimpleGrid>
|
||||
);
|
||||
};
|
||||
|
||||
export async function getServerSideProps(context: NextPageContext) {
|
||||
const page = Number(context.query.page || "1");
|
||||
const blueprints = await getMostRecentBlueprintPages(page);
|
||||
|
||||
return {
|
||||
props: {
|
||||
page,
|
||||
blueprints,
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
export default Index;
|
||||
|
@ -1,355 +0,0 @@
|
||||
import { css } from "@emotion/core";
|
||||
|
||||
export const normalize = css`
|
||||
/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */
|
||||
|
||||
/* Document
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* 1. Correct the line height in all browsers.
|
||||
* 2. Prevent adjustments of font size after orientation changes in iOS.
|
||||
*/
|
||||
|
||||
html {
|
||||
line-height: 1.15; /* 1 */
|
||||
-webkit-text-size-adjust: 100%; /* 2 */
|
||||
}
|
||||
|
||||
/* Sections
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Remove the margin in all browsers.
|
||||
*/
|
||||
|
||||
body {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Render the \`main\` element consistently in IE.
|
||||
*/
|
||||
|
||||
main {
|
||||
display: block;
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the font size and margin on \`h1\` elements within \`section\` and
|
||||
* \`article\` contexts in Chrome, Firefox, and Safari.
|
||||
*/
|
||||
|
||||
h1 {
|
||||
font-size: 2em;
|
||||
margin: 0.67em 0;
|
||||
}
|
||||
|
||||
/* Grouping content
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* 1. Add the correct box sizing in Firefox.
|
||||
* 2. Show the overflow in Edge and IE.
|
||||
*/
|
||||
|
||||
hr {
|
||||
box-sizing: content-box; /* 1 */
|
||||
height: 0; /* 1 */
|
||||
overflow: visible; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the inheritance and scaling of font size in all browsers.
|
||||
* 2. Correct the odd \`em\` font sizing in all browsers.
|
||||
*/
|
||||
|
||||
pre {
|
||||
font-family: monospace, monospace; /* 1 */
|
||||
font-size: 1em; /* 2 */
|
||||
}
|
||||
|
||||
/* Text-level semantics
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Remove the gray background on active links in IE 10.
|
||||
*/
|
||||
|
||||
a {
|
||||
background-color: transparent;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Remove the bottom border in Chrome 57-
|
||||
* 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari.
|
||||
*/
|
||||
|
||||
abbr[title] {
|
||||
border-bottom: none; /* 1 */
|
||||
text-decoration: underline; /* 2 */
|
||||
text-decoration: underline dotted; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct font weight in Chrome, Edge, and Safari.
|
||||
*/
|
||||
|
||||
b,
|
||||
strong {
|
||||
font-weight: bolder;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the inheritance and scaling of font size in all browsers.
|
||||
* 2. Correct the odd \`em\` font sizing in all browsers.
|
||||
*/
|
||||
|
||||
code,
|
||||
kbd,
|
||||
samp {
|
||||
font-family: monospace, monospace; /* 1 */
|
||||
font-size: 1em; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct font size in all browsers.
|
||||
*/
|
||||
|
||||
small {
|
||||
font-size: 80%;
|
||||
}
|
||||
|
||||
/**
|
||||
* Prevent \`sub\` and \`sup\` elements from affecting the line height in
|
||||
* all browsers.
|
||||
*/
|
||||
|
||||
sub,
|
||||
sup {
|
||||
font-size: 75%;
|
||||
line-height: 0;
|
||||
position: relative;
|
||||
vertical-align: baseline;
|
||||
}
|
||||
|
||||
sub {
|
||||
bottom: -0.25em;
|
||||
}
|
||||
|
||||
sup {
|
||||
top: -0.5em;
|
||||
}
|
||||
|
||||
/* Embedded content
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Remove the border on images inside links in IE 10.
|
||||
*/
|
||||
|
||||
img {
|
||||
border-style: none;
|
||||
}
|
||||
|
||||
/* Forms
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* 1. Change the font styles in all browsers.
|
||||
* 2. Remove the margin in Firefox and Safari.
|
||||
*/
|
||||
|
||||
button,
|
||||
input,
|
||||
optgroup,
|
||||
select,
|
||||
textarea {
|
||||
font-family: inherit; /* 1 */
|
||||
font-size: 100%; /* 1 */
|
||||
line-height: 1.15; /* 1 */
|
||||
margin: 0; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the overflow in IE.
|
||||
* 1. Show the overflow in Edge.
|
||||
*/
|
||||
|
||||
button,
|
||||
input {
|
||||
/* 1 */
|
||||
overflow: visible;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the inheritance of text transform in Edge, Firefox, and IE.
|
||||
* 1. Remove the inheritance of text transform in Firefox.
|
||||
*/
|
||||
|
||||
button,
|
||||
select {
|
||||
/* 1 */
|
||||
text-transform: none;
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the inability to style clickable types in iOS and Safari.
|
||||
*/
|
||||
|
||||
button,
|
||||
[type="button"],
|
||||
[type="reset"],
|
||||
[type="submit"] {
|
||||
-webkit-appearance: button;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the inner border and padding in Firefox.
|
||||
*/
|
||||
|
||||
button::-moz-focus-inner,
|
||||
[type="button"]::-moz-focus-inner,
|
||||
[type="reset"]::-moz-focus-inner,
|
||||
[type="submit"]::-moz-focus-inner {
|
||||
border-style: none;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Restore the focus styles unset by the previous rule.
|
||||
*/
|
||||
|
||||
button:-moz-focusring,
|
||||
[type="button"]:-moz-focusring,
|
||||
[type="reset"]:-moz-focusring,
|
||||
[type="submit"]:-moz-focusring {
|
||||
outline: 1px dotted ButtonText;
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the padding in Firefox.
|
||||
*/
|
||||
|
||||
fieldset {
|
||||
padding: 0.35em 0.75em 0.625em;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the text wrapping in Edge and IE.
|
||||
* 2. Correct the color inheritance from \`fieldset\` elements in IE.
|
||||
* 3. Remove the padding so developers are not caught out when they zero out
|
||||
* \`fieldset\` elements in all browsers.
|
||||
*/
|
||||
|
||||
legend {
|
||||
box-sizing: border-box; /* 1 */
|
||||
color: inherit; /* 2 */
|
||||
display: table; /* 1 */
|
||||
max-width: 100%; /* 1 */
|
||||
padding: 0; /* 3 */
|
||||
white-space: normal; /* 1 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct vertical alignment in Chrome, Firefox, and Opera.
|
||||
*/
|
||||
|
||||
progress {
|
||||
vertical-align: baseline;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the default vertical scrollbar in IE 10+.
|
||||
*/
|
||||
|
||||
textarea {
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Add the correct box sizing in IE 10.
|
||||
* 2. Remove the padding in IE 10.
|
||||
*/
|
||||
|
||||
[type="checkbox"],
|
||||
[type="radio"] {
|
||||
box-sizing: border-box; /* 1 */
|
||||
padding: 0; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Correct the cursor style of increment and decrement buttons in Chrome.
|
||||
*/
|
||||
|
||||
[type="number"]::-webkit-inner-spin-button,
|
||||
[type="number"]::-webkit-outer-spin-button {
|
||||
height: auto;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the odd appearance in Chrome and Safari.
|
||||
* 2. Correct the outline style in Safari.
|
||||
*/
|
||||
|
||||
[type="search"] {
|
||||
-webkit-appearance: textfield; /* 1 */
|
||||
outline-offset: -2px; /* 2 */
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the inner padding in Chrome and Safari on macOS.
|
||||
*/
|
||||
|
||||
[type="search"]::-webkit-search-decoration {
|
||||
-webkit-appearance: none;
|
||||
}
|
||||
|
||||
/**
|
||||
* 1. Correct the inability to style clickable types in iOS and Safari.
|
||||
* 2. Change font properties to \`inherit\` in Safari.
|
||||
*/
|
||||
|
||||
::-webkit-file-upload-button {
|
||||
-webkit-appearance: button; /* 1 */
|
||||
font: inherit; /* 2 */
|
||||
}
|
||||
|
||||
/* Interactive
|
||||
========================================================================== */
|
||||
|
||||
/*
|
||||
* Add the correct display in Edge, IE 10+, and Firefox.
|
||||
*/
|
||||
|
||||
details {
|
||||
display: block;
|
||||
}
|
||||
|
||||
/*
|
||||
* Add the correct display in all browsers.
|
||||
*/
|
||||
|
||||
summary {
|
||||
display: list-item;
|
||||
}
|
||||
|
||||
/* Misc
|
||||
========================================================================== */
|
||||
|
||||
/**
|
||||
* Add the correct display in IE 10+.
|
||||
*/
|
||||
|
||||
template {
|
||||
display: none;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the correct display in IE 10.
|
||||
*/
|
||||
|
||||
[hidden] {
|
||||
display: none;
|
||||
}
|
||||
`;
|
@ -1,11 +1,22 @@
|
||||
{
|
||||
"dependencies": {
|
||||
"@emotion/core": "10.0.28",
|
||||
"@chakra-ui/core": "0.8.0",
|
||||
"@emotion/core": "10.0.35",
|
||||
"@emotion/styled": "10.0.27",
|
||||
"document-register-element": "1.13.1",
|
||||
"@google-cloud/datastore": "6.2.0",
|
||||
"@google-cloud/pubsub": "2.6.0",
|
||||
"@google-cloud/storage": "5.3.0",
|
||||
"bbcode-to-react": "0.2.9",
|
||||
"document-register-element": "1.14.10",
|
||||
"emotion-server": "10.0.27",
|
||||
"next": "9.5.2",
|
||||
"emotion-theming": "10.0.27",
|
||||
"next": "9.5.5",
|
||||
"nprogress": "0.2.0",
|
||||
"pako": "1.0.11",
|
||||
"phin": "3.5.0",
|
||||
"react": "16.13.1",
|
||||
"react-dom": "16.13.1"
|
||||
"react-dom": "16.13.1",
|
||||
"react-map-interaction": "2.0.0",
|
||||
"react-markdown": "5.0.1"
|
||||
}
|
||||
}
|
||||
|
63
apps/blueprints/src/BookChildTree.tsx
Normal file
63
apps/blueprints/src/BookChildTree.tsx
Normal file
@ -0,0 +1,63 @@
|
||||
/** @jsx jsx */
|
||||
/* eslint-disable jsx-a11y/anchor-is-valid */
|
||||
import { jsx, css } from "@emotion/core";
|
||||
import Link from "next/link";
|
||||
import BBCode from "bbcode-to-react";
|
||||
import { BlueprintBookEntry } from "@factorio-sites/database";
|
||||
|
||||
const componentStyles = css`
|
||||
.blueprint,
|
||||
.book {
|
||||
display: block;
|
||||
color: white;
|
||||
padding: 4px 0;
|
||||
&:hover {
|
||||
background: #636363;
|
||||
}
|
||||
}
|
||||
.active {
|
||||
background: #41a73a;
|
||||
color: #000;
|
||||
}
|
||||
`;
|
||||
|
||||
interface BookChildTreeProps {
|
||||
child_tree: BlueprintBookEntry["child_tree"];
|
||||
base_url: string;
|
||||
selected_id: string;
|
||||
}
|
||||
|
||||
export const BookChildTree: React.FC<BookChildTreeProps> = ({
|
||||
child_tree,
|
||||
base_url,
|
||||
selected_id,
|
||||
}) => {
|
||||
return (
|
||||
<div css={componentStyles}>
|
||||
{child_tree.map((bp, key) => {
|
||||
return bp.type === "blueprint" ? (
|
||||
<Link key={bp.id} href={`${base_url}?selected=${bp.id}`}>
|
||||
<a className={"blueprint" + (selected_id === bp.id ? " active" : "")}>
|
||||
{BBCode.toReact(bp.name)}
|
||||
</a>
|
||||
</Link>
|
||||
) : (
|
||||
<div key={key}>
|
||||
<Link key={bp.id} href={`${base_url}?selected=${bp.id}&type=book`}>
|
||||
<a className={"book" + (selected_id === bp.id ? " active" : "")}>
|
||||
[book] {BBCode.toReact(bp.name)}
|
||||
</a>
|
||||
</Link>
|
||||
<div css={{ marginLeft: `20px` }}>
|
||||
<BookChildTree
|
||||
child_tree={bp.children}
|
||||
base_url={base_url}
|
||||
selected_id={selected_id}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
);
|
||||
};
|
35
apps/blueprints/src/CopyButton.tsx
Normal file
35
apps/blueprints/src/CopyButton.tsx
Normal file
@ -0,0 +1,35 @@
|
||||
import React, { useState } from "react";
|
||||
import { Button, ButtonProps } from "@chakra-ui/core";
|
||||
|
||||
export const CopyButton: React.FC<Omit<ButtonProps, "children"> & { content: string }> = ({
|
||||
content,
|
||||
...props
|
||||
}) => {
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [icon, setIcon] = useState<"check" | "small-close" | null>(null);
|
||||
|
||||
return (
|
||||
<Button
|
||||
{...props}
|
||||
isLoading={loading}
|
||||
leftIcon={icon}
|
||||
variantColor={icon === "small-close" ? "red" : "green"}
|
||||
onClick={() => {
|
||||
setLoading(true);
|
||||
navigator.clipboard
|
||||
.writeText(content)
|
||||
.then(() => {
|
||||
setLoading(false);
|
||||
setIcon("check");
|
||||
setTimeout(() => setIcon(null), 2500);
|
||||
})
|
||||
.catch(() => {
|
||||
setLoading(false);
|
||||
setIcon("small-close");
|
||||
});
|
||||
}}
|
||||
>
|
||||
copy
|
||||
</Button>
|
||||
);
|
||||
};
|
54
apps/blueprints/src/FullscreenImage.tsx
Normal file
54
apps/blueprints/src/FullscreenImage.tsx
Normal file
@ -0,0 +1,54 @@
|
||||
/** @jsx jsx */
|
||||
/* eslint-disable jsx-a11y/anchor-is-valid */
|
||||
import React, { useState } from "react";
|
||||
import { jsx, css } from "@emotion/core";
|
||||
import { MapInteractionCSS } from "react-map-interaction";
|
||||
|
||||
const elementStyle = css`
|
||||
display: flex;
|
||||
position: fixed;
|
||||
top: 0;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
padding: 0px;
|
||||
background: rgba(0, 0, 0, 0.5);
|
||||
justify-content: center;
|
||||
& > div > div {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
}
|
||||
`;
|
||||
|
||||
interface FullscreenImageProps {
|
||||
alt: string;
|
||||
src: string;
|
||||
close: () => void;
|
||||
}
|
||||
|
||||
export const FullscreenImage: React.FC<FullscreenImageProps> = ({ alt, src, close }) => {
|
||||
const [state, setState] = useState({
|
||||
scale: 0.9,
|
||||
translation: { x: window.innerWidth * 0.05, y: 30 },
|
||||
});
|
||||
|
||||
return (
|
||||
<div
|
||||
css={elementStyle}
|
||||
onClick={(e) => {
|
||||
if ((e as any).target.nodeName.toUpperCase() !== "IMG") {
|
||||
close();
|
||||
}
|
||||
}}
|
||||
onTouchEnd={(e) => {
|
||||
if ((e as any).target.nodeName.toUpperCase() !== "IMG") {
|
||||
close();
|
||||
}
|
||||
}}
|
||||
>
|
||||
<MapInteractionCSS value={state} onChange={setState}>
|
||||
<img alt={alt} src={src} />
|
||||
</MapInteractionCSS>
|
||||
</div>
|
||||
);
|
||||
};
|
61
apps/blueprints/src/Header.tsx
Normal file
61
apps/blueprints/src/Header.tsx
Normal file
@ -0,0 +1,61 @@
|
||||
import React from "react";
|
||||
import { Box, Heading, Flex } from "@chakra-ui/core";
|
||||
import Link from "next/link";
|
||||
|
||||
// const MenuItems = ({ children }) => (
|
||||
// <Text mt={{ base: 4, md: 0 }} mr={6} display="block">
|
||||
// {children}
|
||||
// </Text>
|
||||
// );
|
||||
|
||||
export const Header: React.FC = (props) => {
|
||||
const [show, setShow] = React.useState(false);
|
||||
const handleToggle = () => setShow(!show);
|
||||
|
||||
return (
|
||||
<Flex
|
||||
as="header"
|
||||
align="center"
|
||||
justify="space-between"
|
||||
wrap="wrap"
|
||||
padding="1.5rem"
|
||||
bg="rgba(0,0,0,.7)"
|
||||
color="#ffe6c0"
|
||||
{...props}
|
||||
>
|
||||
<Flex align="center" mr={5}>
|
||||
<Heading as="h1" size="lg">
|
||||
<Link href="/">
|
||||
{/* eslint-disable-next-line jsx-a11y/anchor-is-valid*/}
|
||||
<a>Factorio Blueprints</a>
|
||||
</Link>
|
||||
</Heading>
|
||||
</Flex>
|
||||
|
||||
<Box display={{ base: "block", md: "none" }} onClick={handleToggle}>
|
||||
<svg fill="white" width="12px" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg">
|
||||
<title>Menu</title>
|
||||
<path d="M0 3h20v2H0V3zm0 6h20v2H0V9zm0 6h20v2H0v-2z" />
|
||||
</svg>
|
||||
</Box>
|
||||
|
||||
{/* <Box
|
||||
as="nav"
|
||||
display={{ sm: show ? "block" : "none", md: "flex" }}
|
||||
width={{ sm: "full", md: "auto" }}
|
||||
alignItems="center"
|
||||
flexGrow={1}
|
||||
>
|
||||
<MenuItems>Docs</MenuItems>
|
||||
<MenuItems>Examples</MenuItems>
|
||||
<MenuItems>Blog</MenuItems>
|
||||
</Box> */}
|
||||
|
||||
{/* <Box display={{ sm: show ? "block" : "none", md: "block" }} mt={{ base: 4, md: 0 }}>
|
||||
<Button bg="transparent" border="1px">
|
||||
Create account
|
||||
</Button>
|
||||
</Box> */}
|
||||
</Flex>
|
||||
);
|
||||
};
|
25
apps/blueprints/src/Markdown.tsx
Normal file
25
apps/blueprints/src/Markdown.tsx
Normal file
@ -0,0 +1,25 @@
|
||||
/** @jsx jsx */
|
||||
/* eslint-disable jsx-a11y/anchor-is-valid */
|
||||
import { jsx, css } from "@emotion/core";
|
||||
import ReactMarkdown from "react-markdown";
|
||||
|
||||
const markdownStyle = css`
|
||||
overflow: hidden;
|
||||
img {
|
||||
max-width: 100%;
|
||||
}
|
||||
ul,
|
||||
ol {
|
||||
margin-block-start: 1em;
|
||||
margin-block-end: 1em;
|
||||
margin-inline-start: 0px;
|
||||
margin-inline-end: 0px;
|
||||
padding-inline-start: 40px;
|
||||
}
|
||||
`;
|
||||
|
||||
export const Markdown: React.FC<{ children: string }> = ({ children }) => (
|
||||
<div css={markdownStyle}>
|
||||
<ReactMarkdown>{children}</ReactMarkdown>
|
||||
</div>
|
||||
);
|
35
apps/blueprints/src/Pagination.tsx
Normal file
35
apps/blueprints/src/Pagination.tsx
Normal file
@ -0,0 +1,35 @@
|
||||
/** @jsx jsx */
|
||||
import { jsx } from "@emotion/core";
|
||||
import React from "react";
|
||||
import Link from "next/link";
|
||||
import { useRouter } from "next/router";
|
||||
import { Box, BoxProps, Button } from "@chakra-ui/core";
|
||||
|
||||
interface PaginationProps {
|
||||
page: number;
|
||||
}
|
||||
|
||||
const PaginationLink: React.FC<PaginationProps> = ({ page }) => {
|
||||
const router = useRouter();
|
||||
const query = { ...router.query, page };
|
||||
const href =
|
||||
"?" +
|
||||
Object.keys(query)
|
||||
.map((key) => `${key}=${query[key]}`)
|
||||
.join("&");
|
||||
|
||||
return (
|
||||
<Link href={href} passHref>
|
||||
<Button as="a" size="sm" color="black" css={{ marginRight: "1rem" }}>
|
||||
{page}
|
||||
</Button>
|
||||
</Link>
|
||||
);
|
||||
};
|
||||
|
||||
export const Pagination: React.FC<BoxProps & PaginationProps> = ({ page, ...props }) => (
|
||||
<Box {...props}>
|
||||
{page > 1 && <PaginationLink page={page - 1} />}
|
||||
<PaginationLink page={page + 1} />
|
||||
</Box>
|
||||
);
|
39
apps/blueprints/src/Panel.tsx
Normal file
39
apps/blueprints/src/Panel.tsx
Normal file
@ -0,0 +1,39 @@
|
||||
/** @jsx jsx */
|
||||
import { jsx, css, SerializedStyles } from "@emotion/core";
|
||||
import { Box, BoxProps } from "@chakra-ui/core";
|
||||
import { ReactNode } from "react";
|
||||
|
||||
const panelStyles = css`
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 12px;
|
||||
background-color: #313031;
|
||||
overflow: hidden;
|
||||
box-shadow: inset 3px 0 3px -3px #201815, inset 2px 0 2px -2px #201815,
|
||||
inset 1px 0 1px -1px #201815, inset 0 3px 3px -3px #8f8c8b, inset 0 2px 2px -2px #8f8c8b,
|
||||
inset 0 1px 1px -1px #8f8c8b, inset -3px 0 3px -3px #201815, inset -2px 0 2px -2px #201815,
|
||||
inset -2px 0 1px -1px #201815, inset 0 -3px 3px -3px #000, inset 0 -2px 2px -2px #000,
|
||||
inset 0 -1px 1px -1px #000, 0 0 2px 0 #201815, 0 0 4px 0 #201815;
|
||||
|
||||
h2 {
|
||||
font-weight: 900;
|
||||
color: #ffe6c0;
|
||||
line-height: 1.25;
|
||||
margin: 0 0 12px 0;
|
||||
font-size: 120%;
|
||||
}
|
||||
`;
|
||||
const boxShadow = `inset 0 0 3px 0 #000, 0 -2px 2px -1px #000, -2px 0 2px -2px #28221f,
|
||||
-2px 0 2px -2px #28221f, 2px 0 2px -2px #28221f, 2px 0 2px -2px #28221f,
|
||||
0 3px 3px -3px #8f8c8b, 0 2px 2px -2px #8f8c8b, 0 1px 1px -1px #8f8c8b`;
|
||||
|
||||
export const Panel: React.FC<
|
||||
BoxProps & { title?: string | ReactNode | null; css?: SerializedStyles }
|
||||
> = ({ children, title, css: prop_css, ...props }) => (
|
||||
<Box css={prop_css ? [panelStyles, prop_css] : [panelStyles]} {...props}>
|
||||
{title ? <h2>{title}</h2> : null}
|
||||
<Box color="white" height="100%" padding="12px" bg="#414040" boxShadow={boxShadow}>
|
||||
{children}
|
||||
</Box>
|
||||
</Box>
|
||||
);
|
1
apps/factorioprints-scraper/.eslintrc.json
Normal file
1
apps/factorioprints-scraper/.eslintrc.json
Normal file
@ -0,0 +1 @@
|
||||
{ "extends": "../../.eslintrc.json", "ignorePatterns": ["!**/*"], "rules": {} }
|
97
apps/factorioprints-scraper/data/tags.ts
Normal file
97
apps/factorioprints-scraper/data/tags.ts
Normal file
@ -0,0 +1,97 @@
|
||||
export const tags = {
|
||||
belt: {
|
||||
"0": "balancer",
|
||||
"1": "prioritizer",
|
||||
"2": "tap",
|
||||
"3": "transport belt (yellow)",
|
||||
"4": "fast transport belt (red)",
|
||||
"5": "express transport belt (blue)",
|
||||
},
|
||||
circuit: { "0": "indicator", "1": "counter" },
|
||||
general: {
|
||||
"0": "early game",
|
||||
"1": "mid game",
|
||||
"2": "late game (megabase)",
|
||||
"3": "beaconized",
|
||||
"4": "tileable",
|
||||
"5": "compact",
|
||||
"6": "marathon",
|
||||
"7": "storage",
|
||||
"8": "chunk aligned",
|
||||
},
|
||||
moderation: {
|
||||
"0": "scheduled for deletion",
|
||||
"1": "screenshot missing alt info",
|
||||
"2": "uncropped screenshot",
|
||||
"3": "add english translation",
|
||||
},
|
||||
mods: {
|
||||
"0": "angels",
|
||||
"1": "bobs",
|
||||
"2": "creative",
|
||||
"3": "factorissimo",
|
||||
"4": "warehousing",
|
||||
"5": "lighted-electric-poles",
|
||||
"6": "other",
|
||||
"7": "vanilla",
|
||||
"8": "ltn",
|
||||
"9": "aai",
|
||||
"10": "5dim",
|
||||
"11": "seablock",
|
||||
"12": "industrial revolution",
|
||||
"13": "krastorio",
|
||||
"14": "transport drones",
|
||||
},
|
||||
power: {
|
||||
"0": "nuclear",
|
||||
"1": "kovarex enrichment",
|
||||
"2": "solar",
|
||||
"3": "steam",
|
||||
"4": "accumulator",
|
||||
},
|
||||
production: {
|
||||
"0": "oil processing",
|
||||
"1": "coal liquification",
|
||||
"2": "electronic circuit (green)",
|
||||
"3": "advanced circuit (red)",
|
||||
"4": "processing unit (blue)",
|
||||
"5": "batteries",
|
||||
"6": "rocket parts",
|
||||
"7": "science",
|
||||
"8": "research (labs)",
|
||||
"9": "belts",
|
||||
"10": "smelting",
|
||||
"11": "mining",
|
||||
"12": "uranium",
|
||||
"13": "plastic",
|
||||
"14": "modules",
|
||||
"15": "mall (make everything)",
|
||||
"16": "inserters",
|
||||
"17": "guns and ammo",
|
||||
"18": "robots",
|
||||
"19": "other",
|
||||
"20": "belt based",
|
||||
"21": "logistic (bot) based",
|
||||
},
|
||||
train: {
|
||||
"0": "loading station",
|
||||
"1": "unloading station",
|
||||
"2": "pax",
|
||||
"3": "junction",
|
||||
"4": "roundabout",
|
||||
"5": "crossing",
|
||||
"6": "stacker",
|
||||
"7": "track",
|
||||
"8": "left-hand-drive",
|
||||
"9": "right-hand-drive",
|
||||
},
|
||||
version: {
|
||||
"0": "0,14",
|
||||
"1": "0,15",
|
||||
"2": "0,16",
|
||||
"3": "unknown",
|
||||
"4": "0,17",
|
||||
"5": "0,18",
|
||||
"6": "1,0",
|
||||
},
|
||||
};
|
14
apps/factorioprints-scraper/jest.config.js
Normal file
14
apps/factorioprints-scraper/jest.config.js
Normal file
@ -0,0 +1,14 @@
|
||||
module.exports = {
|
||||
displayName: "factorioprints-scraper",
|
||||
preset: "../../jest.preset.js",
|
||||
globals: {
|
||||
"ts-jest": {
|
||||
tsConfig: "<rootDir>/tsconfig.spec.json",
|
||||
},
|
||||
},
|
||||
transform: {
|
||||
"^.+\\.[tj]s$": "ts-jest",
|
||||
},
|
||||
moduleFileExtensions: ["ts", "js", "html"],
|
||||
coverageDirectory: "../../coverage/apps/factorioprints-scraper",
|
||||
};
|
0
apps/factorioprints-scraper/src/app/.gitkeep
Normal file
0
apps/factorioprints-scraper/src/app/.gitkeep
Normal file
45
apps/factorioprints-scraper/src/app/populate-db.ts
Normal file
45
apps/factorioprints-scraper/src/app/populate-db.ts
Normal file
@ -0,0 +1,45 @@
|
||||
import * as fs from "fs";
|
||||
import * as path from "path";
|
||||
import { promisify } from "util";
|
||||
import {
|
||||
getBlueprintPageByFactorioprintsId,
|
||||
saveBlueprintFromFactorioprints,
|
||||
} from "@factorio-sites/database";
|
||||
|
||||
const fsReadFile = promisify(fs.readFile);
|
||||
const CACHE_DIR = path.join(__dirname, "../../../.cache");
|
||||
|
||||
export async function writeToDatastore() {
|
||||
const filecontent = await fsReadFile(path.join(CACHE_DIR, `most-fav-json/page1.json`), "utf8");
|
||||
const data = JSON.parse(filecontent);
|
||||
|
||||
for (let i = 0; i < data.length; i++) {
|
||||
const bpfilecontent = await fsReadFile(
|
||||
path.join(CACHE_DIR, `blueprint/${data[i].factorioprintsId}.json`),
|
||||
"utf8"
|
||||
);
|
||||
const bp = JSON.parse(bpfilecontent);
|
||||
|
||||
const exists = await getBlueprintPageByFactorioprintsId(bp.factorioprintsId);
|
||||
if (exists) {
|
||||
console.log(`entity ${bp.factorioprintsId} already exists`);
|
||||
continue;
|
||||
}
|
||||
|
||||
await saveBlueprintFromFactorioprints(
|
||||
{
|
||||
title: bp.title,
|
||||
factorioprints_id: bp.factorioprintsId,
|
||||
description_markdown: bp.descriptionMarkdown,
|
||||
tags: bp.tags ? Object.values(bp.tags) : [],
|
||||
updated_at: bp.lastUpdatedDate / 1000,
|
||||
created_at: bp.createdDate / 1000,
|
||||
},
|
||||
bp.blueprintString
|
||||
).catch((reason) => {
|
||||
console.error(reason);
|
||||
});
|
||||
|
||||
console.log(`saved entity ${bp.factorioprintsId} to database`);
|
||||
}
|
||||
}
|
165
apps/factorioprints-scraper/src/app/scan.ts
Normal file
165
apps/factorioprints-scraper/src/app/scan.ts
Normal file
@ -0,0 +1,165 @@
|
||||
import * as crypto from "crypto";
|
||||
import * as fs from "fs";
|
||||
import * as path from "path";
|
||||
import { promisify } from "util";
|
||||
import * as WebSocket from "ws";
|
||||
|
||||
const fsWriteFile = promisify(fs.writeFile);
|
||||
const fsExists = promisify(fs.exists);
|
||||
const CACHE_DIR = path.join(__dirname, "../../../.cache");
|
||||
|
||||
const getMostFavoriteRequestObject = ({
|
||||
lastId,
|
||||
lastNumOfFavorites,
|
||||
}: {
|
||||
lastId?: string;
|
||||
lastNumOfFavorites?: number;
|
||||
}) => ({
|
||||
t: "d",
|
||||
d: {
|
||||
r: 2,
|
||||
a: "q",
|
||||
b: {
|
||||
p: "/blueprintSummaries",
|
||||
q: { ep: lastNumOfFavorites, en: lastId, l: 61, vf: "r", i: "numberOfFavorites" },
|
||||
t: 1,
|
||||
h: "",
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const getBlueprintRequestObject = ({ id }: { id: string }) => ({
|
||||
t: "d",
|
||||
d: { r: 7, a: "q", b: { p: `/blueprints/${id}`, h: "" } },
|
||||
});
|
||||
|
||||
const openWebsocket = (): Promise<WebSocket> => {
|
||||
const ws = new WebSocket("wss://s-usc1c-nss-239.firebaseio.com/.ws?v=5&ns=facorio-blueprints");
|
||||
return new Promise((resolve) => {
|
||||
ws.on("open", function open() {
|
||||
resolve(ws);
|
||||
});
|
||||
ws.on("error", (error) => {
|
||||
console.log("[ws:error]", error);
|
||||
});
|
||||
ws.on("close", () => {
|
||||
console.log("[ws:close]");
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
export async function scanFactorioPrints() {
|
||||
const ws = await openWebsocket();
|
||||
|
||||
const sendMessage = (data: any) => {
|
||||
const string = JSON.stringify(data);
|
||||
console.log("[ws:send]", string);
|
||||
ws.send(string);
|
||||
};
|
||||
|
||||
const sendMessageAndWaitForResponse = (
|
||||
sent_data: any,
|
||||
filter: (data: any) => boolean
|
||||
): Promise<any> => {
|
||||
return new Promise((resolve) => {
|
||||
const buffer = [] as string[];
|
||||
const onmessage = (_message: WebSocket.Data) => {
|
||||
const message = _message.toString();
|
||||
let data;
|
||||
try {
|
||||
data = JSON.parse(message);
|
||||
} catch (e) {
|
||||
buffer.push(message);
|
||||
|
||||
if (buffer.length > 1) {
|
||||
try {
|
||||
data = JSON.parse(buffer.join(""));
|
||||
console.log(`\nValid json found after reading ${buffer.length} messages`);
|
||||
} catch (e) {
|
||||
/**/
|
||||
}
|
||||
}
|
||||
if (!data) return process.stdout.write(buffer.length === 1 ? "Buffering messages." : ".");
|
||||
}
|
||||
|
||||
if (filter(data)) {
|
||||
ws.off("message", onmessage);
|
||||
resolve(data);
|
||||
}
|
||||
};
|
||||
ws.on("message", onmessage);
|
||||
sendMessage(sent_data);
|
||||
});
|
||||
};
|
||||
|
||||
const getMostFavorited = async (
|
||||
page = 1,
|
||||
reqObj: Parameters<typeof getMostFavoriteRequestObject>[0]
|
||||
) => {
|
||||
const data = await sendMessageAndWaitForResponse(
|
||||
getMostFavoriteRequestObject(reqObj),
|
||||
(message) => message?.d?.b?.p === "blueprintSummaries" && message.d.b?.d
|
||||
);
|
||||
|
||||
const _blueprints = data.d.b.d;
|
||||
const ids = Object.keys(_blueprints);
|
||||
|
||||
const blueprints = ids
|
||||
.map((id) => ({
|
||||
factorioprintsId: id,
|
||||
..._blueprints[id],
|
||||
}))
|
||||
.sort((a, b) => b.numberOfFavorites - a.numberOfFavorites);
|
||||
|
||||
// First fetch details of each individual blueprint
|
||||
for (let i = 0; i < blueprints.length; i++) {
|
||||
const blueprint = blueprints[i];
|
||||
const file_path = path.join(CACHE_DIR, `blueprint/${blueprint.factorioprintsId}.json`);
|
||||
|
||||
if (await fsExists(file_path)) {
|
||||
console.log(`Blueprint ${blueprint.factorioprintsId} exists, skipping...`);
|
||||
continue;
|
||||
}
|
||||
|
||||
const result = await sendMessageAndWaitForResponse(
|
||||
getBlueprintRequestObject({ id: blueprint.factorioprintsId }),
|
||||
(data) => data?.d?.b?.p === "blueprints/" + blueprint.factorioprintsId
|
||||
);
|
||||
|
||||
const bp = result.d.b.d;
|
||||
const hash = crypto.createHash("sha1").update(bp.blueprintString).digest("hex");
|
||||
bp.factorioprintsId = blueprint.factorioprintsId;
|
||||
bp.hash = hash;
|
||||
blueprint.hash = hash;
|
||||
|
||||
console.log(`Writing blueprint /blueprint/${bp.factorioprintsId}.json`);
|
||||
|
||||
fsWriteFile(
|
||||
path.join(CACHE_DIR, `blueprint/${bp.factorioprintsId}.json`),
|
||||
JSON.stringify(bp, null, 2)
|
||||
).catch((reason) => {
|
||||
console.error(reason);
|
||||
});
|
||||
}
|
||||
|
||||
// Write the result of entire page including the hashes
|
||||
console.log(`Writing blueprints /most-fav-json/page${page}.json`);
|
||||
|
||||
fsWriteFile(
|
||||
path.join(CACHE_DIR, `most-fav-json/page${page}.json`),
|
||||
JSON.stringify(blueprints, null, 2)
|
||||
).catch((reason) => {
|
||||
console.error(reason);
|
||||
});
|
||||
|
||||
if (page < 3) {
|
||||
const lastBp = blueprints[blueprints.length - 1];
|
||||
getMostFavorited(page + 1, {
|
||||
lastId: lastBp.id,
|
||||
lastNumOfFavorites: lastBp.numberOfFavorites,
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
getMostFavorited(1, {});
|
||||
}
|
0
apps/factorioprints-scraper/src/assets/.gitkeep
Normal file
0
apps/factorioprints-scraper/src/assets/.gitkeep
Normal file
@ -0,0 +1,3 @@
|
||||
export const environment = {
|
||||
production: true,
|
||||
};
|
@ -0,0 +1,3 @@
|
||||
export const environment = {
|
||||
production: false,
|
||||
};
|
32
apps/factorioprints-scraper/src/main.ts
Normal file
32
apps/factorioprints-scraper/src/main.ts
Normal file
@ -0,0 +1,32 @@
|
||||
import { writeToDatastore } from "./app/populate-db";
|
||||
|
||||
// async function writeTestBP() {
|
||||
// const source =
|
||||
// "0eNrtkmFrgzAQhv/KuM+xNBq1zV8pQ9QdW5gmYtIykfz3mZRhh2VLpfsw6Me75J73eO8doWqO2PVCmqJS6h34OHc08MNF6d5EreS5rcWrLBvXM0OHwEEYbIGALFtXaYPYRPUbagOWgJAv+AGc2mcCKI0wAs8YXwyFPLYV9tOHqwACndLTjJJOb+JEyX6TEhiAs3yTWkf18vxiWwJNWeG0IXjIE506J+y1p8Q7ynK2z7OcbrM0mzfcWjL+V0923zwhC1D8d+bGD3PXmbsEJaGg7BcQCwXlN5+bBZ77CivyV5+BrnS4sjbihMXXaAifhvJ9PFcJPPJ6v7zeHrPkvjHzvBUxiIMF6EKA/iBgPwEPcWsm";
|
||||
// const { hash, string } = await parseBlueprintString(source);
|
||||
|
||||
// console.log(`saving ${hash}`);
|
||||
|
||||
// // use createBlueprint
|
||||
// saveBlueprintFromFactorioprints(
|
||||
// {
|
||||
// factorioprints_id: null,
|
||||
// title: "my blueprint",
|
||||
// description_markdown: "",
|
||||
// tags: [],
|
||||
// updated_at: Date.now() / 1000,
|
||||
// created_at: Date.now() / 1000,
|
||||
// },
|
||||
// string
|
||||
// );
|
||||
// }
|
||||
|
||||
async function main() {
|
||||
// scanFactorioPrints();
|
||||
writeToDatastore();
|
||||
// writeTestBP();
|
||||
}
|
||||
|
||||
main().catch((reason) => {
|
||||
console.error(reason);
|
||||
});
|
9
apps/factorioprints-scraper/tsconfig.app.json
Normal file
9
apps/factorioprints-scraper/tsconfig.app.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"types": ["node"]
|
||||
},
|
||||
"exclude": ["**/*.spec.ts"],
|
||||
"include": ["**/*.ts"]
|
||||
}
|
13
apps/factorioprints-scraper/tsconfig.json
Normal file
13
apps/factorioprints-scraper/tsconfig.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"extends": "../../tsconfig.base.json",
|
||||
"files": [],
|
||||
"include": [],
|
||||
"references": [
|
||||
{
|
||||
"path": "./tsconfig.app.json"
|
||||
},
|
||||
{
|
||||
"path": "./tsconfig.spec.json"
|
||||
}
|
||||
]
|
||||
}
|
9
apps/factorioprints-scraper/tsconfig.spec.json
Normal file
9
apps/factorioprints-scraper/tsconfig.spec.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"module": "commonjs",
|
||||
"types": ["jest", "node"]
|
||||
},
|
||||
"include": ["**/*.spec.ts", "**/*.d.ts"]
|
||||
}
|
@ -14,6 +14,7 @@ RUN yarn nx build blueprints
|
||||
FROM node:14-slim
|
||||
|
||||
WORKDIR /usr/src/app
|
||||
|
||||
COPY apps/blueprints/prod.package.json ./package.json
|
||||
COPY yarn.lock .
|
||||
|
||||
|
@ -3,5 +3,10 @@ module.exports = {
|
||||
"<rootDir>/apps/blueprints",
|
||||
"<rootDir>/libs/generate-bp-image",
|
||||
"<rootDir>/apps/blueprint-image-function",
|
||||
"<rootDir>/apps/factorioprints-scraper",
|
||||
"<rootDir>/libs/database",
|
||||
"<rootDir>/libs/utils",
|
||||
"<rootDir>/libs/common-utils",
|
||||
"<rootDir>/libs/web-utils",
|
||||
],
|
||||
};
|
||||
|
1
libs/common-utils/.eslintrc.json
Normal file
1
libs/common-utils/.eslintrc.json
Normal file
@ -0,0 +1 @@
|
||||
{ "extends": "../../.eslintrc.json", "ignorePatterns": ["!**/*"], "rules": {} }
|
7
libs/common-utils/README.md
Normal file
7
libs/common-utils/README.md
Normal file
@ -0,0 +1,7 @@
|
||||
# common-utils
|
||||
|
||||
This library was generated with [Nx](https://nx.dev).
|
||||
|
||||
## Running unit tests
|
||||
|
||||
Run `nx test common-utils` to execute the unit tests via [Jest](https://jestjs.io).
|
14
libs/common-utils/jest.config.js
Normal file
14
libs/common-utils/jest.config.js
Normal file
@ -0,0 +1,14 @@
|
||||
module.exports = {
|
||||
displayName: "common-utils",
|
||||
preset: "../../jest.preset.js",
|
||||
globals: {
|
||||
"ts-jest": {
|
||||
tsConfig: "<rootDir>/tsconfig.spec.json",
|
||||
},
|
||||
},
|
||||
transform: {
|
||||
"^.+\\.[tj]sx?$": "ts-jest",
|
||||
},
|
||||
moduleFileExtensions: ["ts", "tsx", "js", "jsx"],
|
||||
coverageDirectory: "../../coverage/libs/common-utils",
|
||||
};
|
1
libs/common-utils/src/index.ts
Normal file
1
libs/common-utils/src/index.ts
Normal file
@ -0,0 +1 @@
|
||||
export * from "./lib/common-utils";
|
7
libs/common-utils/src/lib/common-utils.spec.ts
Normal file
7
libs/common-utils/src/lib/common-utils.spec.ts
Normal file
@ -0,0 +1,7 @@
|
||||
// import { commonUtils } from "./common-utils";
|
||||
|
||||
describe("commonUtils", () => {
|
||||
it("should work", () => {
|
||||
// expect(commonUtils()).toEqual("common-utils");
|
||||
});
|
||||
});
|
108
libs/common-utils/src/lib/common-utils.ts
Normal file
108
libs/common-utils/src/lib/common-utils.ts
Normal file
@ -0,0 +1,108 @@
|
||||
interface Entity {
|
||||
entity_number: number;
|
||||
name: string;
|
||||
position: { x: number; y: number };
|
||||
}
|
||||
|
||||
export interface Blueprint {
|
||||
entities: Entity[];
|
||||
tiles?: { name: string; position: { x: number; y: number } }[];
|
||||
icons: { signal: { type: "item" | "fluid"; name: string } }[];
|
||||
item: string;
|
||||
label: string;
|
||||
description?: string;
|
||||
version: number;
|
||||
}
|
||||
|
||||
export interface BlueprintBook {
|
||||
active_index: number;
|
||||
blueprints: Array<{ index: number } & BlueprintData>;
|
||||
item: string;
|
||||
label: string;
|
||||
description?: string;
|
||||
version: number;
|
||||
}
|
||||
|
||||
export interface BlueprintData {
|
||||
blueprint_book?: BlueprintBook;
|
||||
blueprint?: Blueprint;
|
||||
}
|
||||
|
||||
export const getBlueprintContentForImageHash = (blueprint: Blueprint): string => {
|
||||
return JSON.stringify({
|
||||
entities: blueprint.entities,
|
||||
tiles: blueprint.tiles,
|
||||
});
|
||||
};
|
||||
|
||||
export const flattenBlueprintData = (data: BlueprintData) => {
|
||||
const blueprints: Blueprint[] = [];
|
||||
const books: BlueprintBook[] = [];
|
||||
|
||||
// Recursively go through the string to find all blueprints
|
||||
const findAndPushBlueprints = (data: BlueprintData) => {
|
||||
if (data.blueprint) {
|
||||
blueprints.push(data.blueprint);
|
||||
} else if (data.blueprint_book) {
|
||||
books.push(data.blueprint_book);
|
||||
data.blueprint_book.blueprints.forEach(({ index, ...bp }) => {
|
||||
findAndPushBlueprints(bp);
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
findAndPushBlueprints(data);
|
||||
|
||||
return {
|
||||
blueprints,
|
||||
books,
|
||||
};
|
||||
};
|
||||
|
||||
export const findBlueprintByPath = (data: BlueprintData, path: number[]): Blueprint | null => {
|
||||
if (path.length === 0) {
|
||||
return (data.blueprint || data.blueprint_book?.blueprints[0]) as Blueprint;
|
||||
} else if (data.blueprint_book && path.length === 1) {
|
||||
return data.blueprint_book.blueprints[path[0]].blueprint as Blueprint;
|
||||
}
|
||||
return null;
|
||||
};
|
||||
|
||||
export const findActiveBlueprint = (
|
||||
data: BlueprintData
|
||||
): { blueprint: Blueprint; path: number[] } => {
|
||||
if (data.blueprint) {
|
||||
return { blueprint: data.blueprint, path: [0] };
|
||||
} else if (data.blueprint_book) {
|
||||
const findActive = (
|
||||
book: BlueprintBook,
|
||||
_path: number[] = []
|
||||
): { blueprint: Blueprint; path: number[] } => {
|
||||
const active = book.blueprints.find((bp) => bp.index === book.active_index);
|
||||
|
||||
if (active && active.blueprint) {
|
||||
return {
|
||||
blueprint: active.blueprint,
|
||||
path: _path.concat(book.active_index),
|
||||
};
|
||||
} else if (active && active.blueprint_book) {
|
||||
return findActive(active.blueprint_book, _path.concat(book.active_index));
|
||||
}
|
||||
|
||||
throw Error("Could not find active blueprint");
|
||||
};
|
||||
|
||||
return findActive(data.blueprint_book);
|
||||
}
|
||||
throw Error("Could not find active blueprint");
|
||||
};
|
||||
|
||||
export const timeLogger = (base_msg: string) => {
|
||||
const start_time = Date.now();
|
||||
let last_time = Date.now();
|
||||
return (message: string) => {
|
||||
const now = Date.now();
|
||||
console.log(`[${base_msg}] ${message} in ${now - last_time} (${now - start_time} total)`);
|
||||
last_time = now;
|
||||
};
|
||||
};
|
13
libs/common-utils/tsconfig.json
Normal file
13
libs/common-utils/tsconfig.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"extends": "../../tsconfig.base.json",
|
||||
"files": [],
|
||||
"include": [],
|
||||
"references": [
|
||||
{
|
||||
"path": "./tsconfig.lib.json"
|
||||
},
|
||||
{
|
||||
"path": "./tsconfig.spec.json"
|
||||
}
|
||||
]
|
||||
}
|
11
libs/common-utils/tsconfig.lib.json
Normal file
11
libs/common-utils/tsconfig.lib.json
Normal file
@ -0,0 +1,11 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"module": "commonjs",
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"declaration": true,
|
||||
"types": ["node"]
|
||||
},
|
||||
"exclude": ["**/*.spec.ts"],
|
||||
"include": ["**/*.ts"]
|
||||
}
|
9
libs/common-utils/tsconfig.spec.json
Normal file
9
libs/common-utils/tsconfig.spec.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"module": "commonjs",
|
||||
"types": ["jest", "node"]
|
||||
},
|
||||
"include": ["**/*.spec.ts", "**/*.spec.tsx", "**/*.spec.js", "**/*.spec.jsx", "**/*.d.ts"]
|
||||
}
|
1
libs/database/.eslintrc.json
Normal file
1
libs/database/.eslintrc.json
Normal file
@ -0,0 +1 @@
|
||||
{ "extends": "../../.eslintrc.json", "ignorePatterns": ["!**/*"], "rules": {} }
|
7
libs/database/README.md
Normal file
7
libs/database/README.md
Normal file
@ -0,0 +1,7 @@
|
||||
# database
|
||||
|
||||
This library was generated with [Nx](https://nx.dev).
|
||||
|
||||
## Running unit tests
|
||||
|
||||
Run `nx test database` to execute the unit tests via [Jest](https://jestjs.io).
|
14
libs/database/jest.config.js
Normal file
14
libs/database/jest.config.js
Normal file
@ -0,0 +1,14 @@
|
||||
module.exports = {
|
||||
displayName: "database",
|
||||
preset: "../../jest.preset.js",
|
||||
globals: {
|
||||
"ts-jest": {
|
||||
tsConfig: "<rootDir>/tsconfig.spec.json",
|
||||
},
|
||||
},
|
||||
transform: {
|
||||
"^.+\\.[tj]sx?$": "ts-jest",
|
||||
},
|
||||
moduleFileExtensions: ["ts", "tsx", "js", "jsx"],
|
||||
coverageDirectory: "../../coverage/libs/database",
|
||||
};
|
2
libs/database/src/index.ts
Normal file
2
libs/database/src/index.ts
Normal file
@ -0,0 +1,2 @@
|
||||
export * from "./lib/database";
|
||||
export * from "./lib/pubsub";
|
7
libs/database/src/lib/database.spec.ts
Normal file
7
libs/database/src/lib/database.spec.ts
Normal file
@ -0,0 +1,7 @@
|
||||
import { database } from "./database";
|
||||
|
||||
describe("database", () => {
|
||||
it("should work", () => {
|
||||
expect(database()).toEqual("database");
|
||||
});
|
||||
});
|
547
libs/database/src/lib/database.ts
Normal file
547
libs/database/src/lib/database.ts
Normal file
@ -0,0 +1,547 @@
|
||||
import { Storage } from "@google-cloud/storage";
|
||||
import { Datastore } from "@google-cloud/datastore";
|
||||
import { encodeBlueprint, hashString, parseBlueprintString } from "@factorio-sites/node-utils";
|
||||
import {
|
||||
Blueprint,
|
||||
BlueprintBook,
|
||||
getBlueprintContentForImageHash,
|
||||
} from "@factorio-sites/common-utils";
|
||||
import { getBlueprintImageRequestTopic } from "./pubsub";
|
||||
|
||||
// to dev on windows run: $env:GOOGLE_APPLICATION_CREDENTIALS="FULL_PATH"
|
||||
const storage = new Storage();
|
||||
const datastore = new Datastore();
|
||||
|
||||
const BLUEPRINT_BUCKET = storage.bucket("blueprint-strings");
|
||||
const IMAGE_BUCKET = storage.bucket("blueprint-images");
|
||||
const BlueprintEntity = "Blueprint";
|
||||
const BlueprintBookEntity = "BlueprintBook";
|
||||
const BlueprintPageEntity = "BlueprintPage";
|
||||
const BlueprintStringEntity = "BlueprintString";
|
||||
const blueprintImageRequestTopic = getBlueprintImageRequestTopic();
|
||||
|
||||
interface BlueprintChild {
|
||||
type: "blueprint";
|
||||
id: string;
|
||||
name: string;
|
||||
}
|
||||
interface BlueprintBookChild {
|
||||
type: "blueprint_book";
|
||||
id: string;
|
||||
name: string;
|
||||
children: ChildTree;
|
||||
}
|
||||
type ChildTree = Array<BlueprintChild | BlueprintBookChild>;
|
||||
|
||||
export interface BlueprintEntry {
|
||||
id: string;
|
||||
label: string; // from source
|
||||
description: string | null; // from source
|
||||
game_version: number; // from source
|
||||
blueprint_hash: string;
|
||||
image_hash: string;
|
||||
created_at: number;
|
||||
updated_at: number;
|
||||
tags: string[];
|
||||
factorioprints_id?: string;
|
||||
// BlueprintEntry->BlueprintString 1:m
|
||||
// BlueprintEntry->BlueprintPageEntry n:m
|
||||
}
|
||||
|
||||
export interface BlueprintBookEntry {
|
||||
id: string;
|
||||
label: string;
|
||||
description?: string;
|
||||
/** strings as keys of BlueprintEntry */
|
||||
blueprint_ids: string[];
|
||||
/** strings as keys of BlueprintBookEntry (currently unsupported) */
|
||||
blueprint_book_ids: string[];
|
||||
child_tree: ChildTree;
|
||||
blueprint_hash: string;
|
||||
created_at: number;
|
||||
updated_at: number;
|
||||
is_modded: boolean;
|
||||
factorioprints_id?: string;
|
||||
// BlueprintBook:BlueprintBook n:m
|
||||
// BlueprintBook:BlueprintEntry 1:m
|
||||
}
|
||||
|
||||
export interface BlueprintPageEntry {
|
||||
id: string;
|
||||
blueprint_id?: string;
|
||||
blueprint_book_id?: string;
|
||||
title: string;
|
||||
description_markdown: string;
|
||||
created_at: number;
|
||||
updated_at: number;
|
||||
factorioprints_id?: string;
|
||||
// BlueprintPageEntry->BlueprintEntry 1:m
|
||||
// BlueprintPageEntry->BlueprintBook 1:m
|
||||
}
|
||||
|
||||
export interface BlueprintStringEntry {
|
||||
blueprint_id: string;
|
||||
blueprint_hash: string;
|
||||
image_hash: string;
|
||||
version: number;
|
||||
changes_markdown: string;
|
||||
created_at: Date;
|
||||
// BlueprintString->BlueprintEntry m:1
|
||||
}
|
||||
|
||||
class DatastoreExistsError extends Error {
|
||||
constructor(public existingId: string, message: string) {
|
||||
super(message);
|
||||
this.name = this.constructor.name;
|
||||
Error.captureStackTrace(this, this.constructor);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* BlueprintBook
|
||||
*/
|
||||
|
||||
const mapBlueprintBookEntityToObject = (entity: any): BlueprintBookEntry => ({
|
||||
id: entity[datastore.KEY].id,
|
||||
blueprint_ids: entity.blueprint_ids.map((key: any) => key.id),
|
||||
blueprint_book_ids: entity.blueprint_book_ids.map((key: any) => key.id),
|
||||
child_tree: entity.child_tree ? entity.child_tree : [],
|
||||
blueprint_hash: entity.blueprint_hash,
|
||||
label: entity.label,
|
||||
description: entity.description,
|
||||
created_at: entity.created_at && entity.created_at.getTime() / 1000,
|
||||
updated_at: entity.updated_at && entity.updated_at.getTime() / 1000,
|
||||
is_modded: entity.is_modded || false,
|
||||
factorioprints_id: entity.factorioprints_id || null,
|
||||
});
|
||||
|
||||
export async function getBlueprintBookById(id: string): Promise<BlueprintBookEntry | null> {
|
||||
const result = await datastore.get(datastore.key([BlueprintBookEntity, Number(id)]));
|
||||
return result[0] ? mapBlueprintBookEntityToObject(result[0]) : null;
|
||||
}
|
||||
|
||||
export async function getBlueprintBookByHash(hash: string): Promise<BlueprintBookEntry | null> {
|
||||
const query = datastore
|
||||
.createQuery(BlueprintBookEntity)
|
||||
.filter("blueprint_hash", "=", hash)
|
||||
.limit(1);
|
||||
const [result] = await datastore.runQuery(query);
|
||||
return result[0] ? mapBlueprintBookEntityToObject(result[0]) : null;
|
||||
}
|
||||
|
||||
async function createBlueprintBook(
|
||||
blueprintBook: BlueprintBook,
|
||||
extraInfo: { tags: string[]; created_at: number; updated_at: number; factorioprints_id: string }
|
||||
): Promise<{ insertedId: string; child_tree: ChildTree }> {
|
||||
const string = await encodeBlueprint({ blueprint_book: blueprintBook });
|
||||
const blueprint_hash = hashString(string);
|
||||
|
||||
const exists = await getBlueprintBookByHash(blueprint_hash);
|
||||
if (exists) {
|
||||
throw new DatastoreExistsError(exists.id, "Blueprint book already exists");
|
||||
}
|
||||
|
||||
// Write string to google storage
|
||||
await BLUEPRINT_BUCKET.file(blueprint_hash).save(string);
|
||||
|
||||
const blueprint_ids = [];
|
||||
const blueprint_book_ids = [];
|
||||
const child_tree: ChildTree = [];
|
||||
|
||||
// Create the book's child objects
|
||||
for (let i = 0; i < blueprintBook.blueprints.length; i++) {
|
||||
const blueprint = blueprintBook.blueprints[i];
|
||||
if (blueprint.blueprint) {
|
||||
const result = await createBlueprint(blueprint.blueprint, extraInfo).catch((error) => {
|
||||
if (error instanceof DatastoreExistsError) {
|
||||
console.log(`Blueprint already exists with id ${error.existingId}`);
|
||||
return { insertedId: error.existingId };
|
||||
} else throw error;
|
||||
});
|
||||
child_tree.push({
|
||||
type: "blueprint",
|
||||
id: result.insertedId,
|
||||
name: blueprint.blueprint.label,
|
||||
});
|
||||
blueprint_ids.push(result.insertedId);
|
||||
} else if (blueprint.blueprint_book) {
|
||||
const result = await createBlueprintBook(blueprint.blueprint_book, extraInfo).catch(
|
||||
(error) => {
|
||||
if (error instanceof DatastoreExistsError) {
|
||||
console.log(`Blueprint book already exists with id ${error.existingId}`);
|
||||
// TODO: query blueprint book to get child_tree
|
||||
return { insertedId: error.existingId, child_tree: [] as ChildTree };
|
||||
} else throw error;
|
||||
}
|
||||
);
|
||||
child_tree.push({
|
||||
type: "blueprint_book",
|
||||
id: result.insertedId,
|
||||
name: blueprint.blueprint_book.label,
|
||||
children: result.child_tree,
|
||||
});
|
||||
blueprint_book_ids.push(result.insertedId);
|
||||
}
|
||||
}
|
||||
|
||||
// Write blueprint details to datastore
|
||||
const [result] = await datastore.insert({
|
||||
key: datastore.key([BlueprintBookEntity]),
|
||||
data: [
|
||||
{
|
||||
name: "blueprint_ids",
|
||||
value: blueprint_ids.map((id) => datastore.key([BlueprintEntity, datastore.int(id)])),
|
||||
},
|
||||
{
|
||||
name: "blueprint_book_ids",
|
||||
value: blueprint_book_ids.map((id) =>
|
||||
datastore.key([BlueprintBookEntity, datastore.int(id)])
|
||||
),
|
||||
},
|
||||
{ name: "child_tree", value: child_tree, excludeFromIndexes: true },
|
||||
{ name: "label", value: blueprintBook.label },
|
||||
{
|
||||
name: "description",
|
||||
value: blueprintBook.description || null,
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{ name: "blueprint_hash", value: blueprint_hash },
|
||||
{
|
||||
name: "created_at",
|
||||
value: extraInfo.created_at ? new Date(extraInfo.created_at * 1000) : null,
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{
|
||||
name: "updated_at",
|
||||
value: extraInfo.updated_at ? new Date(extraInfo.updated_at * 1000) : null,
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{ name: "factorioprints_id", value: extraInfo.factorioprints_id || null },
|
||||
],
|
||||
});
|
||||
|
||||
const insertedId = String(result.mutationResults?.[0]?.key?.path?.[0]?.id) as string;
|
||||
if (!insertedId) throw Error("Something went wrong inserting entity");
|
||||
|
||||
console.log(`Created Blueprint book ${insertedId}`);
|
||||
|
||||
return { insertedId, child_tree };
|
||||
}
|
||||
|
||||
/*
|
||||
* Blueprint
|
||||
*/
|
||||
|
||||
const mapBlueprintEntityToObject = (entity: any): BlueprintEntry => ({
|
||||
id: entity[datastore.KEY].id,
|
||||
blueprint_hash: entity.blueprint_hash,
|
||||
image_hash: entity.image_hash,
|
||||
label: entity.label,
|
||||
description: entity.description,
|
||||
tags: entity.tags,
|
||||
created_at: entity.created_at && entity.created_at.getTime() / 1000,
|
||||
updated_at: entity.updated_at && entity.updated_at.getTime() / 1000,
|
||||
factorioprints_id: entity.factorioprints_id || null,
|
||||
game_version: entity.game_version,
|
||||
});
|
||||
|
||||
const mapBlueprintObjectToEntity = (object: Omit<BlueprintEntry, "id"> & { id?: string }): any => ({
|
||||
key: datastore.key(object.id ? [BlueprintEntity, datastore.int(object.id)] : [BlueprintEntity]),
|
||||
data: [
|
||||
{ name: "label", value: object.label },
|
||||
{
|
||||
name: "description",
|
||||
value: object.description || null,
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{ name: "blueprint_hash", value: object.blueprint_hash },
|
||||
{ name: "image_hash", value: object.image_hash },
|
||||
{ name: "tags", value: object.tags || [] },
|
||||
{
|
||||
name: "created_at",
|
||||
value: object.created_at ? new Date(object.created_at * 1000) : new Date(),
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{
|
||||
name: "updated_at",
|
||||
value: object.updated_at ? new Date(object.updated_at * 1000) : new Date(),
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{ name: "game_version", value: object.game_version, excludeFromIndexes: true },
|
||||
{ name: "factorioprints_id", value: object.factorioprints_id || null },
|
||||
],
|
||||
});
|
||||
|
||||
export async function getBlueprintById(id: string): Promise<BlueprintEntry | null> {
|
||||
const result = await datastore.get(datastore.key([BlueprintEntity, datastore.int(id)]));
|
||||
return result[0] ? mapBlueprintEntityToObject(result[0]) : null;
|
||||
}
|
||||
|
||||
export async function getBlueprintByHash(hash: string): Promise<BlueprintEntry | null> {
|
||||
const query = datastore.createQuery(BlueprintEntity).filter("blueprint_hash", "=", hash).limit(1);
|
||||
const [result] = await datastore.runQuery(query);
|
||||
return result[0] ? mapBlueprintEntityToObject(result[0]) : null;
|
||||
}
|
||||
|
||||
export async function getBlueprintByImageHash(hash: string): Promise<BlueprintEntry | null> {
|
||||
const query = datastore.createQuery(BlueprintEntity).filter("image_hash", "=", hash).limit(1);
|
||||
const [result] = await datastore.runQuery(query);
|
||||
return result[0] ? mapBlueprintEntityToObject(result[0]) : null;
|
||||
}
|
||||
|
||||
export async function createBlueprint(
|
||||
blueprint: Blueprint,
|
||||
extraInfo: { tags: string[]; created_at: number; updated_at: number; factorioprints_id: string }
|
||||
) {
|
||||
const string = await encodeBlueprint({ blueprint });
|
||||
const blueprint_hash = hashString(string);
|
||||
const image_hash = hashString(getBlueprintContentForImageHash(blueprint));
|
||||
|
||||
const exists = await getBlueprintByHash(blueprint_hash);
|
||||
if (exists) {
|
||||
throw new DatastoreExistsError(exists.id, "Blueprint already exists");
|
||||
}
|
||||
|
||||
// Write string to google storage
|
||||
await BLUEPRINT_BUCKET.file(blueprint_hash).save(string);
|
||||
|
||||
// Write blueprint details to datastore
|
||||
const [result] = await datastore.insert({
|
||||
key: datastore.key([BlueprintEntity]),
|
||||
data: [
|
||||
{ name: "label", value: blueprint.label },
|
||||
{
|
||||
name: "description",
|
||||
value: blueprint.description || null,
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{ name: "blueprint_hash", value: blueprint_hash },
|
||||
{ name: "image_hash", value: image_hash },
|
||||
{ name: "tags", value: extraInfo.tags ? extraInfo.tags : [] },
|
||||
{
|
||||
name: "created_at",
|
||||
value: extraInfo.created_at ? new Date(extraInfo.created_at * 1000) : new Date(),
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{
|
||||
name: "updated_at",
|
||||
value: extraInfo.updated_at ? new Date(extraInfo.updated_at * 1000) : new Date(),
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{ name: "game_version", value: blueprint.version, excludeFromIndexes: true },
|
||||
{ name: "factorioprints_id", value: extraInfo.factorioprints_id || null },
|
||||
],
|
||||
});
|
||||
|
||||
const insertedId = String(result.mutationResults?.[0]?.key?.path?.[0]?.id);
|
||||
if (!insertedId) throw Error("Something went wrong inserting entity");
|
||||
|
||||
console.log(`Created Blueprint ${insertedId}`);
|
||||
|
||||
blueprintImageRequestTopic.publishJSON({
|
||||
blueprintId: insertedId,
|
||||
});
|
||||
|
||||
await datastore.insert({
|
||||
key: datastore.key([BlueprintEntity, datastore.int(insertedId), BlueprintStringEntity]),
|
||||
data: [
|
||||
{ name: "blueprint_hash", value: blueprint_hash },
|
||||
{ name: "image_hash", value: image_hash },
|
||||
{ name: "version", value: 1 },
|
||||
{ name: "changes_markdown", value: null },
|
||||
{
|
||||
name: "created_at",
|
||||
value: extraInfo.created_at ? new Date(extraInfo.created_at * 1000) : new Date(),
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
],
|
||||
});
|
||||
|
||||
return { insertedId };
|
||||
}
|
||||
|
||||
export async function updateBlueprint(blueprint: BlueprintEntry) {
|
||||
datastore.save(mapBlueprintObjectToEntity(blueprint));
|
||||
}
|
||||
|
||||
/*
|
||||
* BlueprintString
|
||||
*/
|
||||
|
||||
export async function getBlueprintStringByHash(hash: string): Promise<string | null> {
|
||||
const [buffer] = await BLUEPRINT_BUCKET.file(hash).download();
|
||||
return buffer ? buffer.toString() : null;
|
||||
}
|
||||
|
||||
/*
|
||||
* BlueprintPage
|
||||
*/
|
||||
const mapBlueprintPageEntityToObject = (entity: any): BlueprintPageEntry => ({
|
||||
id: entity[datastore.KEY].id,
|
||||
blueprint_id: entity.blueprint_id ? entity.blueprint_id.id : null,
|
||||
blueprint_book_id: entity.blueprint_book_id ? entity.blueprint_book_id.id : null,
|
||||
title: entity.title,
|
||||
description_markdown: entity.description_markdown,
|
||||
created_at: entity.created_at && entity.created_at.getTime() / 1000,
|
||||
updated_at: entity.updated_at && entity.updated_at.getTime() / 1000,
|
||||
factorioprints_id: entity.factorioprints_id || null,
|
||||
});
|
||||
|
||||
export async function getBlueprintPageById(id: string): Promise<BlueprintPageEntry | null> {
|
||||
const result = await datastore.get(datastore.key([BlueprintPageEntity, datastore.int(id)]));
|
||||
return result[0] ? mapBlueprintPageEntityToObject(result[0]) : null;
|
||||
}
|
||||
|
||||
export async function getBlueprintPageByFactorioprintsId(
|
||||
id: string
|
||||
): Promise<BlueprintPageEntry | null> {
|
||||
const query = datastore
|
||||
.createQuery(BlueprintPageEntity)
|
||||
.filter("factorioprints_id", "=", id)
|
||||
.limit(1);
|
||||
const [result] = await datastore.runQuery(query);
|
||||
return result[0] ? mapBlueprintPageEntityToObject(result[0]) : null;
|
||||
}
|
||||
|
||||
async function createBlueprintPage(
|
||||
type: "blueprint" | "blueprint_book",
|
||||
targetId: string,
|
||||
extraInfo: {
|
||||
title: string;
|
||||
description_markdown: string;
|
||||
created_at: number;
|
||||
updated_at: number;
|
||||
factorioprints_id?: string;
|
||||
}
|
||||
) {
|
||||
const insertData: any = [
|
||||
{ name: "title", value: extraInfo.title },
|
||||
{
|
||||
name: "description_markdown",
|
||||
value: extraInfo.description_markdown,
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{
|
||||
name: "created_at",
|
||||
value: extraInfo.created_at ? new Date(extraInfo.created_at * 1000) : new Date(),
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{
|
||||
name: "updated_at",
|
||||
value: extraInfo.updated_at ? new Date(extraInfo.updated_at * 1000) : new Date(),
|
||||
excludeFromIndexes: true,
|
||||
},
|
||||
{ name: "factorioprints_id", value: extraInfo.factorioprints_id || null },
|
||||
];
|
||||
|
||||
if (type === "blueprint") {
|
||||
insertData.push({
|
||||
name: "blueprint_id",
|
||||
value: datastore.key([BlueprintEntity, datastore.int(targetId)]),
|
||||
});
|
||||
} else if (type === "blueprint_book") {
|
||||
insertData.push({
|
||||
name: "blueprint_book_id",
|
||||
value: datastore.key([BlueprintBookEntity, datastore.int(targetId)]),
|
||||
});
|
||||
} else {
|
||||
throw Error("Invalid type given");
|
||||
}
|
||||
|
||||
await datastore.insert({
|
||||
key: datastore.key([BlueprintPageEntity]),
|
||||
data: insertData,
|
||||
});
|
||||
|
||||
console.log(`Created Blueprint Page`);
|
||||
}
|
||||
|
||||
/*
|
||||
* BlueprintImage
|
||||
*/
|
||||
|
||||
export async function saveBlueprintImage(hash: string, image: Buffer): Promise<void> {
|
||||
return IMAGE_BUCKET.file(`${hash}.webp`).save(image, {
|
||||
contentType: "image/webp",
|
||||
});
|
||||
}
|
||||
|
||||
export async function hasBlueprintImage(hash: string): Promise<boolean> {
|
||||
const [result] = await IMAGE_BUCKET.file(`${hash}.webp`).exists();
|
||||
return result;
|
||||
}
|
||||
|
||||
/*
|
||||
* Other
|
||||
*/
|
||||
|
||||
interface BlueprintDataFromFactorioprints {
|
||||
description_markdown: string;
|
||||
title: string;
|
||||
updated_at: number;
|
||||
created_at: number;
|
||||
tags: string[];
|
||||
factorioprints_id: string;
|
||||
}
|
||||
export async function saveBlueprintFromFactorioprints(
|
||||
factorioprintData: BlueprintDataFromFactorioprints,
|
||||
blueprintString: string
|
||||
) {
|
||||
const parsed = await parseBlueprintString(blueprintString);
|
||||
|
||||
// not needed for inserting, just printing
|
||||
// const { blueprints, books } = flattenBlueprintData(parsed.data);
|
||||
// console.log(`string has ${books.length} books with ${blueprints.length} blueprints`);
|
||||
|
||||
const extraInfo = {
|
||||
created_at: factorioprintData.created_at,
|
||||
updated_at: factorioprintData.updated_at,
|
||||
tags: factorioprintData.tags,
|
||||
factorioprints_id: factorioprintData.factorioprints_id,
|
||||
};
|
||||
|
||||
const extraInfoPage = {
|
||||
title: factorioprintData.title,
|
||||
description_markdown: factorioprintData.description_markdown,
|
||||
created_at: factorioprintData.created_at,
|
||||
updated_at: factorioprintData.updated_at,
|
||||
factorioprints_id: factorioprintData.factorioprints_id,
|
||||
};
|
||||
|
||||
if (parsed.data.blueprint) {
|
||||
console.log(`string has one blueprint...`);
|
||||
const { insertedId } = await createBlueprint(parsed.data.blueprint, extraInfo).catch(
|
||||
(error) => {
|
||||
if (error instanceof DatastoreExistsError) {
|
||||
console.log(`Blueprint already exists with id ${error.existingId}`);
|
||||
return { insertedId: error.existingId };
|
||||
} else throw error;
|
||||
}
|
||||
);
|
||||
await createBlueprintPage("blueprint", insertedId, extraInfoPage);
|
||||
} else if (parsed.data.blueprint_book) {
|
||||
console.log(`string has a blueprint book...`);
|
||||
const { insertedId } = await createBlueprintBook(parsed.data.blueprint_book, extraInfo);
|
||||
await createBlueprintPage("blueprint_book", insertedId, extraInfoPage);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
export async function getMostRecentBlueprintPages(page = 1): Promise<BlueprintPageEntry[]> {
|
||||
const perPage = 10;
|
||||
const query = datastore
|
||||
.createQuery(BlueprintPageEntity)
|
||||
.limit(perPage)
|
||||
.offset((page - 1) * perPage);
|
||||
const [result] = await datastore.runQuery(query);
|
||||
return result.map(mapBlueprintPageEntityToObject);
|
||||
}
|
||||
|
||||
export async function getPaginatedBlueprints(page = 1, perPage = 10): Promise<BlueprintEntry[]> {
|
||||
const query = datastore
|
||||
.createQuery(BlueprintEntity)
|
||||
.limit(perPage)
|
||||
.offset((page - 1) * perPage);
|
||||
const [result] = await datastore.runQuery(query);
|
||||
return result.map(mapBlueprintEntityToObject);
|
||||
}
|
10
libs/database/src/lib/pubsub.ts
Normal file
10
libs/database/src/lib/pubsub.ts
Normal file
@ -0,0 +1,10 @@
|
||||
import { PubSub, Message } from "@google-cloud/pubsub";
|
||||
// export { Message } from "@google-cloud/pubsub";
|
||||
|
||||
const pubsub = new PubSub();
|
||||
|
||||
export function getBlueprintImageRequestTopic() {
|
||||
return pubsub.topic("projects/factorio-sites/topics/blueprint-image-request");
|
||||
}
|
||||
|
||||
export type PubSubMessage = Message;
|
13
libs/database/tsconfig.json
Normal file
13
libs/database/tsconfig.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"extends": "../../tsconfig.base.json",
|
||||
"files": [],
|
||||
"include": [],
|
||||
"references": [
|
||||
{
|
||||
"path": "./tsconfig.lib.json"
|
||||
},
|
||||
{
|
||||
"path": "./tsconfig.spec.json"
|
||||
}
|
||||
]
|
||||
}
|
11
libs/database/tsconfig.lib.json
Normal file
11
libs/database/tsconfig.lib.json
Normal file
@ -0,0 +1,11 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"module": "commonjs",
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"declaration": true,
|
||||
"types": ["node"]
|
||||
},
|
||||
"exclude": ["**/*.spec.ts"],
|
||||
"include": ["**/*.ts"]
|
||||
}
|
9
libs/database/tsconfig.spec.json
Normal file
9
libs/database/tsconfig.spec.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"module": "commonjs",
|
||||
"types": ["jest", "node"]
|
||||
},
|
||||
"include": ["**/*.spec.ts", "**/*.spec.tsx", "**/*.spec.js", "**/*.spec.jsx", "**/*.d.ts"]
|
||||
}
|
@ -1,54 +1,80 @@
|
||||
import * as Puppeteer from "puppeteer";
|
||||
import * as crypto from "crypto";
|
||||
import * as path from "path";
|
||||
import * as fs from "fs";
|
||||
import { promisify } from "util";
|
||||
import * as Puppeteer from "puppeteer";
|
||||
import { BlueprintEntry } from "@factorio-sites/database";
|
||||
import { timeLogger } from "@factorio-sites/common-utils";
|
||||
|
||||
const fsMkdir = promisify(fs.mkdir);
|
||||
const fsStat = promisify(fs.stat);
|
||||
const fsRmdir = promisify(fs.rmdir);
|
||||
const fsReadFile = promisify(fs.readFile);
|
||||
|
||||
async function downloadScreenshot(blueprint: string, dir: string, on_complete: Promise<any>) {
|
||||
const browser = await Puppeteer.launch({
|
||||
headless: true,
|
||||
args: ["--no-sandbox"],
|
||||
});
|
||||
let browser: Puppeteer.Browser;
|
||||
Puppeteer.launch({
|
||||
headless: true,
|
||||
args: ["--no-sandbox"],
|
||||
}).then((_browser) => (browser = _browser));
|
||||
|
||||
const [page] = await browser.pages();
|
||||
async function downloadScreenshot(
|
||||
blueprint: BlueprintEntry,
|
||||
dir: string,
|
||||
on_complete: Promise<any>
|
||||
) {
|
||||
// const browser = await Puppeteer.launch({
|
||||
// headless: true,
|
||||
// args: ["--no-sandbox"],
|
||||
// });
|
||||
const tl = timeLogger("downloadScreenshot");
|
||||
|
||||
// const [page] = await browser.pages();
|
||||
const page = await browser.newPage();
|
||||
|
||||
await (page as any)._client.send("Page.setDownloadBehavior", {
|
||||
behavior: "allow",
|
||||
downloadPath: dir,
|
||||
});
|
||||
|
||||
const start = Date.now();
|
||||
let pageErroredOnBlueprint = false;
|
||||
|
||||
await page.goto("https://teoxoy.github.io/factorio-blueprint-editor/?source=" + blueprint, {
|
||||
page.on("console", (message) => {
|
||||
if (message.type() === "error" && message.text() === "JSHandle@array")
|
||||
pageErroredOnBlueprint = true;
|
||||
});
|
||||
|
||||
const bp_url = `https://factorio-blueprints-nleb5djksq-ey.a.run.app/api/string/${blueprint.blueprint_hash}`;
|
||||
tl(`Opening web page with blueprint_hash ${blueprint.blueprint_hash}`);
|
||||
await page.goto("https://teoxoy.github.io/factorio-blueprint-editor/?source=" + bp_url, {
|
||||
waitUntil: "load",
|
||||
});
|
||||
console.log(`[status] page load complete at ${Date.now() - start}`);
|
||||
|
||||
tl(`page load complete`);
|
||||
|
||||
await page.waitForFunction(`!!document.querySelector('.toasts-text')`);
|
||||
|
||||
console.log(`[status] app initialized at ${Date.now() - start}`);
|
||||
if (pageErroredOnBlueprint) throw Error("Failed to parse blueprint string");
|
||||
|
||||
await page.click("canvas");
|
||||
tl(`app initialized`);
|
||||
|
||||
await page.focus("canvas");
|
||||
await page.keyboard.down("Control");
|
||||
await page.keyboard.press("S");
|
||||
await page.keyboard.up("Control");
|
||||
|
||||
console.log("[status] save image command entered");
|
||||
tl("save image command entered");
|
||||
|
||||
on_complete.finally(() => {
|
||||
browser.close();
|
||||
// browser.close();
|
||||
page.close();
|
||||
});
|
||||
}
|
||||
|
||||
export async function generateScreenshot(blueprint: string, _cache_dir?: string) {
|
||||
const start_time = Date.now();
|
||||
const hash = crypto.createHash("sha256").update(blueprint).digest("hex");
|
||||
const cache_dir = _cache_dir || path.join(process.cwd(), ".cache");
|
||||
const dir = path.join(cache_dir, hash);
|
||||
export async function generateScreenshot(blueprint: BlueprintEntry, _cache_dir?: string) {
|
||||
const cache_dir = _cache_dir || path.join(process.cwd(), ".cache/image-downloads");
|
||||
const dir = path.join(cache_dir, String(blueprint.id));
|
||||
|
||||
if (!blueprint.blueprint_hash) {
|
||||
throw Error("Failed to generate screenshot, no blueprint hash found");
|
||||
}
|
||||
|
||||
await fsMkdir(cache_dir).catch((error) => {
|
||||
if (error.code !== "EEXIST") throw error;
|
||||
@ -57,6 +83,19 @@ export async function generateScreenshot(blueprint: string, _cache_dir?: string)
|
||||
if (error.code !== "EEXIST") throw error;
|
||||
});
|
||||
|
||||
const result = await _generateScreenshot(blueprint, dir);
|
||||
// .finally(() => {
|
||||
// fsRmdir(dir, { recursive: true, maxRetries: 3, retryDelay: 100 }).catch((reason) => {
|
||||
// console.log(`clearing directory failed: ${reason.code}`);
|
||||
// });
|
||||
// });
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
async function _generateScreenshot(blueprint: BlueprintEntry, dir: string) {
|
||||
const tl = timeLogger("generateScreenshot");
|
||||
|
||||
const promise = new Promise<string>((resolve, reject) => {
|
||||
const watcher = fs.watch(dir, async (type, file) => {
|
||||
if (type === "change" && !file.endsWith(".crdownload")) {
|
||||
@ -64,7 +103,10 @@ export async function generateScreenshot(blueprint: string, _cache_dir?: string)
|
||||
fsStat(file_path)
|
||||
.then(() => resolve(file_path))
|
||||
.catch(reject)
|
||||
.finally(() => watcher.close());
|
||||
.finally(() => {
|
||||
console.log(`closing watcher ${dir}`);
|
||||
watcher.close();
|
||||
});
|
||||
}
|
||||
});
|
||||
});
|
||||
@ -73,11 +115,16 @@ export async function generateScreenshot(blueprint: string, _cache_dir?: string)
|
||||
|
||||
const file_path = await promise;
|
||||
|
||||
console.log(`Downloaded image in ${Date.now() - start_time}ms`, file_path);
|
||||
const stream = fs.createReadStream(file_path);
|
||||
stream.on("close", () => {
|
||||
fsRmdir(dir, { recursive: true }).catch((reason) => console.error(reason));
|
||||
});
|
||||
tl(`Downloaded image ${file_path}`);
|
||||
|
||||
return stream;
|
||||
const buffer = await fsReadFile(file_path);
|
||||
|
||||
const buffermin = buffer;
|
||||
// const buffermin = await imagemin.buffer(buffer, {
|
||||
// plugins: [imageminWebp({ quality: 50 })],
|
||||
// });
|
||||
|
||||
tl("imageminWebp");
|
||||
|
||||
return buffermin;
|
||||
}
|
||||
|
1
libs/node-utils/.eslintrc.json
Normal file
1
libs/node-utils/.eslintrc.json
Normal file
@ -0,0 +1 @@
|
||||
{ "extends": "../../.eslintrc.json", "ignorePatterns": ["!**/*"], "rules": {} }
|
7
libs/node-utils/README.md
Normal file
7
libs/node-utils/README.md
Normal file
@ -0,0 +1,7 @@
|
||||
# node-utils
|
||||
|
||||
This library was generated with [Nx](https://nx.dev).
|
||||
|
||||
## Running unit tests
|
||||
|
||||
Run `nx test node-utils` to execute the unit tests via [Jest](https://jestjs.io).
|
14
libs/node-utils/jest.config.js
Normal file
14
libs/node-utils/jest.config.js
Normal file
@ -0,0 +1,14 @@
|
||||
module.exports = {
|
||||
displayName: "node-utils",
|
||||
preset: "../../jest.preset.js",
|
||||
globals: {
|
||||
"ts-jest": {
|
||||
tsConfig: "<rootDir>/tsconfig.spec.json",
|
||||
},
|
||||
},
|
||||
transform: {
|
||||
"^.+\\.[tj]sx?$": "ts-jest",
|
||||
},
|
||||
moduleFileExtensions: ["ts", "tsx", "js", "jsx"],
|
||||
coverageDirectory: "../../coverage/libs/node-utils",
|
||||
};
|
1
libs/node-utils/src/index.ts
Normal file
1
libs/node-utils/src/index.ts
Normal file
@ -0,0 +1 @@
|
||||
export * from "./lib/node-utils";
|
7
libs/node-utils/src/lib/node-utils.spec.ts
Normal file
7
libs/node-utils/src/lib/node-utils.spec.ts
Normal file
@ -0,0 +1,7 @@
|
||||
// import { utils } from "./node-utils";
|
||||
|
||||
describe("utils", () => {
|
||||
it("should work", () => {
|
||||
// expect(true).toEqual("utils");
|
||||
});
|
||||
});
|
37
libs/node-utils/src/lib/node-utils.ts
Normal file
37
libs/node-utils/src/lib/node-utils.ts
Normal file
@ -0,0 +1,37 @@
|
||||
import * as crypto from "crypto";
|
||||
import * as pako from "pako";
|
||||
// import * as phin from "phin";
|
||||
import { BlueprintData } from "@factorio-sites/common-utils";
|
||||
|
||||
export const parseBlueprintString = async (
|
||||
string: string
|
||||
): Promise<{ hash: string; data: BlueprintData; string: string }> => {
|
||||
// if (string.startsWith("http:") || string.startsWith("https:")) {
|
||||
// const result = await phin(string);
|
||||
// string = result.body.toString();
|
||||
// }
|
||||
|
||||
const hash = crypto.createHash("sha1").update(string).digest("hex");
|
||||
const buffer = Buffer.from(string.substr(1), "base64");
|
||||
const decoded = pako.inflate(buffer);
|
||||
const json = new TextDecoder("utf-8").decode(decoded);
|
||||
const data = JSON.parse(json);
|
||||
|
||||
return {
|
||||
hash,
|
||||
data,
|
||||
string,
|
||||
};
|
||||
};
|
||||
|
||||
export const encodeBlueprint = async (data: BlueprintData): Promise<string> => {
|
||||
const json = JSON.stringify(data);
|
||||
const encoded = new TextEncoder().encode(json);
|
||||
const compressed = pako.deflate(encoded);
|
||||
const base64 = Buffer.from(compressed).toString("base64");
|
||||
return "0" + base64;
|
||||
};
|
||||
|
||||
export const hashString = (string: string) => {
|
||||
return crypto.createHash("sha1").update(string).digest("hex");
|
||||
};
|
13
libs/node-utils/tsconfig.json
Normal file
13
libs/node-utils/tsconfig.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"extends": "../../tsconfig.base.json",
|
||||
"files": [],
|
||||
"include": [],
|
||||
"references": [
|
||||
{
|
||||
"path": "./tsconfig.lib.json"
|
||||
},
|
||||
{
|
||||
"path": "./tsconfig.spec.json"
|
||||
}
|
||||
]
|
||||
}
|
11
libs/node-utils/tsconfig.lib.json
Normal file
11
libs/node-utils/tsconfig.lib.json
Normal file
@ -0,0 +1,11 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"module": "commonjs",
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"declaration": true,
|
||||
"types": ["node"]
|
||||
},
|
||||
"exclude": ["**/*.spec.ts"],
|
||||
"include": ["**/*.ts"]
|
||||
}
|
9
libs/node-utils/tsconfig.spec.json
Normal file
9
libs/node-utils/tsconfig.spec.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"module": "commonjs",
|
||||
"types": ["jest", "node"]
|
||||
},
|
||||
"include": ["**/*.spec.ts", "**/*.spec.tsx", "**/*.spec.js", "**/*.spec.jsx", "**/*.d.ts"]
|
||||
}
|
1
libs/web-utils/.eslintrc.json
Normal file
1
libs/web-utils/.eslintrc.json
Normal file
@ -0,0 +1 @@
|
||||
{ "extends": "../../.eslintrc.json", "ignorePatterns": ["!**/*"], "rules": {} }
|
7
libs/web-utils/README.md
Normal file
7
libs/web-utils/README.md
Normal file
@ -0,0 +1,7 @@
|
||||
# web-utils
|
||||
|
||||
This library was generated with [Nx](https://nx.dev).
|
||||
|
||||
## Running unit tests
|
||||
|
||||
Run `nx test web-utils` to execute the unit tests via [Jest](https://jestjs.io).
|
14
libs/web-utils/jest.config.js
Normal file
14
libs/web-utils/jest.config.js
Normal file
@ -0,0 +1,14 @@
|
||||
module.exports = {
|
||||
displayName: "web-utils",
|
||||
preset: "../../jest.preset.js",
|
||||
globals: {
|
||||
"ts-jest": {
|
||||
tsConfig: "<rootDir>/tsconfig.spec.json",
|
||||
},
|
||||
},
|
||||
transform: {
|
||||
"^.+\\.[tj]sx?$": "ts-jest",
|
||||
},
|
||||
moduleFileExtensions: ["ts", "tsx", "js", "jsx"],
|
||||
coverageDirectory: "../../coverage/libs/web-utils",
|
||||
};
|
1
libs/web-utils/src/index.ts
Normal file
1
libs/web-utils/src/index.ts
Normal file
@ -0,0 +1 @@
|
||||
export * from "./lib/web-utils";
|
28
libs/web-utils/src/lib/web-utils.ts
Normal file
28
libs/web-utils/src/lib/web-utils.ts
Normal file
@ -0,0 +1,28 @@
|
||||
import * as pako from "pako";
|
||||
import { BlueprintData } from "@factorio-sites/common-utils";
|
||||
|
||||
export function parseBlueprintStringClient(source: string): { data: BlueprintData } {
|
||||
const encoded = atob(source.substring(1));
|
||||
const decoded = pako.inflate(encoded);
|
||||
const string = new TextDecoder("utf-8").decode(decoded);
|
||||
const jsonObject = JSON.parse(string);
|
||||
return { data: jsonObject };
|
||||
}
|
||||
|
||||
export function encodeBlueprintStringClient(data: BlueprintData): string {
|
||||
const json = JSON.stringify(data);
|
||||
const encoded = new TextEncoder().encode(json);
|
||||
const compressed = pako.deflate(encoded, { to: "string" });
|
||||
const base64 = btoa(compressed);
|
||||
return "0" + base64;
|
||||
}
|
||||
|
||||
export function chakraResponsive({
|
||||
mobile,
|
||||
desktop,
|
||||
}: {
|
||||
mobile: string;
|
||||
desktop: string;
|
||||
}): string[] {
|
||||
return [mobile, mobile, desktop, desktop];
|
||||
}
|
13
libs/web-utils/tsconfig.json
Normal file
13
libs/web-utils/tsconfig.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"extends": "../../tsconfig.base.json",
|
||||
"files": [],
|
||||
"include": [],
|
||||
"references": [
|
||||
{
|
||||
"path": "./tsconfig.lib.json"
|
||||
},
|
||||
{
|
||||
"path": "./tsconfig.spec.json"
|
||||
}
|
||||
]
|
||||
}
|
9
libs/web-utils/tsconfig.lib.json
Normal file
9
libs/web-utils/tsconfig.lib.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"types": []
|
||||
},
|
||||
"exclude": ["**/*.spec.ts"],
|
||||
"include": ["**/*.ts"]
|
||||
}
|
9
libs/web-utils/tsconfig.spec.json
Normal file
9
libs/web-utils/tsconfig.spec.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"outDir": "../../dist/out-tsc",
|
||||
"module": "commonjs",
|
||||
"types": ["jest", "node"]
|
||||
},
|
||||
"include": ["**/*.spec.ts", "**/*.spec.tsx", "**/*.spec.js", "**/*.spec.jsx", "**/*.d.ts"]
|
||||
}
|
15
nx.json
15
nx.json
@ -35,6 +35,21 @@
|
||||
},
|
||||
"blueprint-image-function": {
|
||||
"tags": []
|
||||
},
|
||||
"factorioprints-scraper": {
|
||||
"tags": []
|
||||
},
|
||||
"database": {
|
||||
"tags": []
|
||||
},
|
||||
"node-utils": {
|
||||
"tags": []
|
||||
},
|
||||
"common-utils": {
|
||||
"tags": []
|
||||
},
|
||||
"web-utils": {
|
||||
"tags": []
|
||||
}
|
||||
}
|
||||
}
|
||||
|
38
package.json
38
package.json
@ -27,14 +27,27 @@
|
||||
},
|
||||
"private": true,
|
||||
"dependencies": {
|
||||
"@emotion/core": "10.0.28",
|
||||
"@chakra-ui/core": "0.8.0",
|
||||
"@emotion/core": "10.0.35",
|
||||
"@emotion/styled": "10.0.27",
|
||||
"document-register-element": "1.13.1",
|
||||
"@google-cloud/datastore": "6.2.0",
|
||||
"@google-cloud/pubsub": "2.6.0",
|
||||
"@google-cloud/storage": "5.3.0",
|
||||
"bbcode-to-react": "0.2.9",
|
||||
"document-register-element": "1.14.10",
|
||||
"emotion-server": "10.0.27",
|
||||
"next": "9.5.2",
|
||||
"emotion-theming": "10.0.27",
|
||||
"next": "9.5.5",
|
||||
"nprogress": "0.2.0",
|
||||
"pako": "1.0.11",
|
||||
"phin": "3.5.0",
|
||||
"puppeteer": "5.3.1",
|
||||
"react": "16.13.1",
|
||||
"react-dom": "16.13.1"
|
||||
"react-dom": "16.13.1",
|
||||
"react-map-interaction": "2.0.0",
|
||||
"react-markdown": "5.0.1",
|
||||
"sharp": "0.26.2",
|
||||
"ws": "7.3.1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@babel/core": "7.9.6",
|
||||
@ -52,25 +65,32 @@
|
||||
"@nrwl/web": "10.3.1",
|
||||
"@nrwl/workspace": "10.3.1",
|
||||
"@testing-library/react": "10.4.1",
|
||||
"@types/bbcode-to-react": "0.2.0",
|
||||
"@types/imagemin": "7.0.0",
|
||||
"@types/imagemin-webp": "5.1.1",
|
||||
"@types/jest": "26.0.8",
|
||||
"@types/node": "14.11.10",
|
||||
"@types/node": "14.14.1",
|
||||
"@types/nprogress": "0.2.0",
|
||||
"@types/pako": "1.0.1",
|
||||
"@types/puppeteer": "3.0.2",
|
||||
"@types/react": "16.9.38",
|
||||
"@types/react": "16.9.53",
|
||||
"@types/react-dom": "16.9.8",
|
||||
"@types/sharp": "0.26.0",
|
||||
"@types/ws": "7.2.7",
|
||||
"@typescript-eslint/eslint-plugin": "4.3.0",
|
||||
"@typescript-eslint/parser": "4.3.0",
|
||||
"babel-jest": "26.2.2",
|
||||
"cypress": "^4.1.0",
|
||||
"cypress": "4.1.0",
|
||||
"dotenv": "6.2.0",
|
||||
"eslint": "7.10.0",
|
||||
"eslint-config-prettier": "6.0.0",
|
||||
"eslint-plugin-cypress": "^2.10.3",
|
||||
"eslint-plugin-cypress": "2.10.3",
|
||||
"eslint-plugin-import": "2.21.2",
|
||||
"eslint-plugin-jsx-a11y": "6.3.1",
|
||||
"eslint-plugin-react": "7.20.0",
|
||||
"eslint-plugin-react-hooks": "4.0.4",
|
||||
"jest": "26.2.2",
|
||||
"prettier": "2.0.4",
|
||||
"prettier": "2.1.2",
|
||||
"ts-jest": "26.4.0",
|
||||
"ts-node": "~7.0.0",
|
||||
"tslint": "~6.0.0",
|
||||
|
@ -17,7 +17,11 @@
|
||||
"strict": true,
|
||||
"baseUrl": ".",
|
||||
"paths": {
|
||||
"@factorio-sites/generate-bp-image": ["libs/generate-bp-image/src/index.ts"]
|
||||
"@factorio-sites/generate-bp-image": ["libs/generate-bp-image/src/index.ts"],
|
||||
"@factorio-sites/database": ["libs/database/src/index.ts"],
|
||||
"@factorio-sites/node-utils": ["libs/node-utils/src/index.ts"],
|
||||
"@factorio-sites/common-utils": ["libs/common-utils/src/index.ts"],
|
||||
"@factorio-sites/web-utils": ["libs/web-utils/src/index.ts"]
|
||||
}
|
||||
},
|
||||
"exclude": ["node_modules", "tmp"]
|
||||
|
134
workspace.json
134
workspace.json
@ -147,6 +147,140 @@
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"factorioprints-scraper": {
|
||||
"root": "apps/factorioprints-scraper",
|
||||
"sourceRoot": "apps/factorioprints-scraper/src",
|
||||
"projectType": "application",
|
||||
"prefix": "factorioprints-scraper",
|
||||
"schematics": {},
|
||||
"architect": {
|
||||
"build": {
|
||||
"builder": "@nrwl/node:build",
|
||||
"options": {
|
||||
"outputPath": "dist/apps/factorioprints-scraper",
|
||||
"main": "apps/factorioprints-scraper/src/main.ts",
|
||||
"tsConfig": "apps/factorioprints-scraper/tsconfig.app.json",
|
||||
"assets": ["apps/factorioprints-scraper/src/assets"]
|
||||
},
|
||||
"configurations": {
|
||||
"production": {
|
||||
"optimization": true,
|
||||
"extractLicenses": true,
|
||||
"inspect": false,
|
||||
"fileReplacements": [
|
||||
{
|
||||
"replace": "apps/factorioprints-scraper/src/environments/environment.ts",
|
||||
"with": "apps/factorioprints-scraper/src/environments/environment.prod.ts"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
"serve": {
|
||||
"builder": "@nrwl/node:execute",
|
||||
"options": {
|
||||
"buildTarget": "factorioprints-scraper:build"
|
||||
}
|
||||
},
|
||||
"lint": {
|
||||
"builder": "@nrwl/linter:eslint",
|
||||
"options": {
|
||||
"lintFilePatterns": ["apps/factorioprints-scraper/**/*.ts"]
|
||||
}
|
||||
},
|
||||
"test": {
|
||||
"builder": "@nrwl/jest:jest",
|
||||
"options": {
|
||||
"jestConfig": "apps/factorioprints-scraper/jest.config.js",
|
||||
"passWithNoTests": true
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"database": {
|
||||
"root": "libs/database",
|
||||
"sourceRoot": "libs/database/src",
|
||||
"projectType": "library",
|
||||
"schematics": {},
|
||||
"architect": {
|
||||
"lint": {
|
||||
"builder": "@nrwl/linter:eslint",
|
||||
"options": {
|
||||
"lintFilePatterns": ["libs/database/**/*.ts"]
|
||||
}
|
||||
},
|
||||
"test": {
|
||||
"builder": "@nrwl/jest:jest",
|
||||
"options": {
|
||||
"jestConfig": "libs/database/jest.config.js",
|
||||
"passWithNoTests": true
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"node-utils": {
|
||||
"root": "libs/node-utils",
|
||||
"sourceRoot": "libs/node-utils/src",
|
||||
"projectType": "library",
|
||||
"schematics": {},
|
||||
"architect": {
|
||||
"lint": {
|
||||
"builder": "@nrwl/linter:eslint",
|
||||
"options": {
|
||||
"lintFilePatterns": ["libs/node-utils/**/*.ts"]
|
||||
}
|
||||
},
|
||||
"test": {
|
||||
"builder": "@nrwl/jest:jest",
|
||||
"options": {
|
||||
"jestConfig": "libs/node-utils/jest.config.js",
|
||||
"passWithNoTests": true
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"common-utils": {
|
||||
"root": "libs/common-utils",
|
||||
"sourceRoot": "libs/common-utils/src",
|
||||
"projectType": "library",
|
||||
"schematics": {},
|
||||
"architect": {
|
||||
"lint": {
|
||||
"builder": "@nrwl/linter:eslint",
|
||||
"options": {
|
||||
"lintFilePatterns": ["libs/common-utils/**/*.ts"]
|
||||
}
|
||||
},
|
||||
"test": {
|
||||
"builder": "@nrwl/jest:jest",
|
||||
"options": {
|
||||
"jestConfig": "libs/common-utils/jest.config.js",
|
||||
"passWithNoTests": true
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"web-utils": {
|
||||
"root": "libs/web-utils",
|
||||
"sourceRoot": "libs/web-utils/src",
|
||||
"projectType": "library",
|
||||
"schematics": {},
|
||||
"architect": {
|
||||
"lint": {
|
||||
"builder": "@nrwl/linter:eslint",
|
||||
"options": {
|
||||
"lintFilePatterns": ["libs/web-utils/**/*.ts"]
|
||||
}
|
||||
},
|
||||
"test": {
|
||||
"builder": "@nrwl/jest:jest",
|
||||
"options": {
|
||||
"jestConfig": "libs/web-utils/jest.config.js",
|
||||
"passWithNoTests": true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"cli": {
|
||||
|
Loading…
x
Reference in New Issue
Block a user