2020-10-06 22:07:43 +03:00
|
|
|
import * as Environment from "~/node_common/environment";
|
|
|
|
import * as Utilities from "~/node_common/utilities";
|
|
|
|
|
|
|
|
import configs from "~/knexfile";
|
|
|
|
import knex from "knex";
|
2020-10-06 23:35:22 +03:00
|
|
|
import FS from "fs";
|
2020-10-06 22:07:43 +03:00
|
|
|
|
|
|
|
import { PrivateKey } from "@textile/hub";
|
2020-10-06 22:27:27 +03:00
|
|
|
import { execSync } from "child_process";
|
2020-10-06 22:07:43 +03:00
|
|
|
|
|
|
|
const envConfig = configs["development"];
|
|
|
|
const db = knex(envConfig);
|
|
|
|
|
|
|
|
// NOTE(jim):
|
|
|
|
// This script will reproduce the bucket bricking bug
|
|
|
|
// We will trigger it by trying to upload a file thats too big.
|
|
|
|
console.log(`RUNNING: recreate-bucket-bug.js`);
|
|
|
|
|
2020-10-06 23:35:22 +03:00
|
|
|
const HIGH_WATER_MARK = 1024 * 1024 * 3;
|
|
|
|
|
2020-10-06 22:07:43 +03:00
|
|
|
const TEXTILE_KEY_INFO = {
|
|
|
|
key: Environment.TEXTILE_HUB_KEY,
|
|
|
|
secret: Environment.TEXTILE_HUB_SECRET,
|
|
|
|
};
|
|
|
|
|
2020-10-06 23:35:22 +03:00
|
|
|
const reportError = (message) => {
|
|
|
|
console.log(
|
|
|
|
`\x1b[1m[ \x1b[31mTEXTILE ERROR\x1b[0m\x1b[1m ]\x1b[0m ${message}`
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const reportTask = (message) => {
|
|
|
|
console.log(`\x1b[1m[ \x1b[32mSCRIPT\x1b[0m\x1b[1m ]\x1b[0m ${message}`);
|
|
|
|
};
|
|
|
|
|
2020-10-06 22:07:43 +03:00
|
|
|
const run = async () => {
|
2020-10-06 23:35:22 +03:00
|
|
|
reportTask(`creating key`);
|
|
|
|
|
2020-10-06 22:07:43 +03:00
|
|
|
const identity = await PrivateKey.fromRandom();
|
|
|
|
const api = identity.toString();
|
|
|
|
|
|
|
|
const user = {
|
|
|
|
username: "SCRIPT_USER",
|
|
|
|
data: {
|
|
|
|
tokens: {
|
|
|
|
api,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2020-10-06 23:35:22 +03:00
|
|
|
reportTask(
|
2020-10-06 23:36:53 +03:00
|
|
|
`initializing bucket with .env variables from your Slate environment`
|
2020-10-06 23:35:22 +03:00
|
|
|
);
|
|
|
|
|
2020-10-06 22:07:43 +03:00
|
|
|
console.log({ api });
|
|
|
|
|
|
|
|
let { buckets, bucketKey } = await Utilities.getBucketAPIFromUserToken({
|
|
|
|
user,
|
|
|
|
bucketName: "delete-this-bucket",
|
|
|
|
});
|
|
|
|
|
|
|
|
// NOTE(jim): Verify this new bucket exists.
|
2020-10-06 23:35:22 +03:00
|
|
|
//
|
|
|
|
// ----------------------------------------------------------------
|
|
|
|
|
|
|
|
reportTask(`attempting to print bucket`);
|
|
|
|
|
2020-10-06 22:07:43 +03:00
|
|
|
let list = null;
|
|
|
|
try {
|
|
|
|
list = await buckets.list();
|
|
|
|
console.log({ list });
|
|
|
|
} catch (e) {
|
2020-10-06 23:35:22 +03:00
|
|
|
reportError(e.message);
|
2020-10-06 22:07:43 +03:00
|
|
|
}
|
|
|
|
|
2020-10-06 22:27:27 +03:00
|
|
|
// NOTE(jim): Create 200MB File
|
|
|
|
// Create 500MB File
|
|
|
|
// Create 1GB File
|
|
|
|
// Create 2GB File
|
|
|
|
// Create 4GB File
|
2020-10-06 23:35:22 +03:00
|
|
|
// ----------------------------------------------------------------
|
|
|
|
|
|
|
|
reportTask(`creating test files, hardcoded in script file`);
|
|
|
|
|
|
|
|
const files = [];
|
2020-10-06 22:27:27 +03:00
|
|
|
try {
|
|
|
|
await execSync(
|
|
|
|
"dd if=/dev/random of=200MB_BUCKET_TEST.txt bs=1 count=0 seek=200m"
|
|
|
|
);
|
2020-10-06 23:35:22 +03:00
|
|
|
files.push("200MB_BUCKET_TEST.txt");
|
|
|
|
|
2020-10-06 22:27:27 +03:00
|
|
|
await execSync(
|
|
|
|
"dd if=/dev/random of=500MB_BUCKET_TEST.txt bs=1 count=0 seek=500m"
|
|
|
|
);
|
2020-10-06 23:35:22 +03:00
|
|
|
files.push("500MB_BUCKET_TEST.txt");
|
|
|
|
|
2020-10-06 22:27:27 +03:00
|
|
|
await execSync(
|
|
|
|
"dd if=/dev/random of=1GB_BUCKET_TEST.txt bs=1 count=0 seek=1g"
|
|
|
|
);
|
2020-10-06 23:35:22 +03:00
|
|
|
files.push("1GB_BUCKET_TEST.txt");
|
|
|
|
|
2020-10-06 22:27:27 +03:00
|
|
|
await execSync(
|
|
|
|
"dd if=/dev/random of=2GB_BUCKET_TEST.txt bs=1 count=0 seek=2g"
|
|
|
|
);
|
2020-10-06 23:35:22 +03:00
|
|
|
files.push("2GB_BUCKET_TEST.txt");
|
|
|
|
|
|
|
|
// NOTE(jim): For Textile
|
|
|
|
// Uncommenting this block will trigger a bug we should
|
|
|
|
// Resolve first.
|
|
|
|
// ----------------------------------------------------------------
|
|
|
|
|
2020-10-06 22:27:27 +03:00
|
|
|
await execSync(
|
|
|
|
"dd if=/dev/random of=4GB_BUCKET_TEST.txt bs=1 count=0 seek=4g"
|
|
|
|
);
|
2020-10-06 23:35:22 +03:00
|
|
|
files.push("4GB_BUCKET_TEST.txt");
|
|
|
|
} catch (e) {
|
|
|
|
reportError(e.message);
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log({ files });
|
|
|
|
|
|
|
|
// NOTE(jim): Try to upload each file the first time.
|
|
|
|
// ----------------------------------------------------------------
|
|
|
|
let items;
|
|
|
|
for (let i = 0; i < files.length; i++) {
|
|
|
|
const file = files[i];
|
|
|
|
|
|
|
|
const readStream = FS.createReadStream(`${file}`, {
|
|
|
|
highWaterMark: HIGH_WATER_MARK,
|
|
|
|
});
|
|
|
|
|
|
|
|
reportTask(`attempting ${file} push to bucket`);
|
|
|
|
|
|
|
|
try {
|
|
|
|
await buckets.pushPath(bucketKey, file, readStream);
|
2020-10-07 00:03:03 +03:00
|
|
|
reportTask(`successfully added ${file}`);
|
2020-10-06 23:35:22 +03:00
|
|
|
} catch (e) {
|
|
|
|
reportError(e.message);
|
|
|
|
}
|
|
|
|
|
|
|
|
items = null;
|
|
|
|
try {
|
|
|
|
reportTask(`attempting listPath for ${list[0].key}`);
|
|
|
|
const response = await buckets.listPath(list[0].key, "/");
|
|
|
|
items = response.item.items;
|
|
|
|
} catch (e) {
|
|
|
|
reportError(e.message);
|
|
|
|
}
|
|
|
|
|
2020-10-07 00:47:16 +03:00
|
|
|
if (items) {
|
|
|
|
reportTask(`there are ${items.length} items, including .textileseed`);
|
|
|
|
} else {
|
|
|
|
reportError(`listPath failed so there are no items to work with`);
|
|
|
|
}
|
2020-10-06 23:35:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE(jim): Remove each file
|
|
|
|
// We remove each file so that we can upload more files again.
|
|
|
|
// ----------------------------------------------------------------
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
|
|
|
const item = items[i];
|
|
|
|
|
2020-10-07 00:47:16 +03:00
|
|
|
reportTask(`attempting to remove ${item.name}`);
|
2020-10-06 23:35:22 +03:00
|
|
|
try {
|
|
|
|
await buckets.removePath(bucketKey, item.name);
|
2020-10-07 00:47:16 +03:00
|
|
|
reportTask(`${item.name} has been removed`);
|
2020-10-06 23:35:22 +03:00
|
|
|
} catch (e) {
|
|
|
|
reportError(e.message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE(jim): here we check to make sure that each bucket item
|
|
|
|
// has been removed appropriately.
|
|
|
|
// ----------------------------------------------------------------
|
|
|
|
try {
|
|
|
|
reportTask(`attempting listPath for ${list[0].key} after deletions`);
|
|
|
|
const response = await buckets.listPath(list[0].key, "/");
|
|
|
|
items = response.item.items;
|
2020-10-06 22:27:27 +03:00
|
|
|
} catch (e) {
|
2020-10-06 23:35:22 +03:00
|
|
|
reportError(e.message);
|
|
|
|
}
|
|
|
|
|
2020-10-07 00:47:16 +03:00
|
|
|
reportTask(`there are ${items.length} items`);
|
2020-10-06 23:35:22 +03:00
|
|
|
|
|
|
|
// NOTE(jim): Try to upload each file again.
|
|
|
|
// This helps us check whether or not the bucket can handle
|
|
|
|
// uploads after files have been removed.
|
|
|
|
// ----------------------------------------------------------------
|
|
|
|
for (let i = 0; i < files.length; i++) {
|
|
|
|
const file = files[i];
|
|
|
|
|
|
|
|
const readStream = FS.createReadStream(`${file}`, {
|
|
|
|
highWaterMark: HIGH_WATER_MARK,
|
|
|
|
});
|
|
|
|
|
|
|
|
reportTask(
|
|
|
|
`\x1b[1m[ second upload phase ]\x1b[0m attempting ${file} push to bucket`
|
|
|
|
);
|
|
|
|
|
|
|
|
try {
|
|
|
|
await buckets.pushPath(bucketKey, file, readStream);
|
2020-10-07 00:03:03 +03:00
|
|
|
reportTask(
|
|
|
|
`\x1b[1m[ second upload phase ]\x1b[0m successfully added ${file}`
|
|
|
|
);
|
2020-10-06 23:35:22 +03:00
|
|
|
} catch (e) {
|
|
|
|
reportError(e.message);
|
|
|
|
}
|
|
|
|
|
|
|
|
items = null;
|
|
|
|
try {
|
|
|
|
reportTask(
|
|
|
|
`\x1b[1m[ second upload phase ]\x1b[0m attempting listPath for ${
|
|
|
|
list[0].key
|
|
|
|
}`
|
|
|
|
);
|
|
|
|
const response = await buckets.listPath(list[0].key, "/");
|
|
|
|
items = response.item.items;
|
|
|
|
} catch (e) {
|
|
|
|
reportError(e.message);
|
|
|
|
}
|
|
|
|
|
2020-10-07 00:47:16 +03:00
|
|
|
if (!items) {
|
|
|
|
reportError(
|
|
|
|
`\x1b[1m[ second upload phase ]\x1b[0m can't report because listPath failed`
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
reportTask(
|
|
|
|
`\x1b[1m[ second upload phase ]\x1b[0m there are ${
|
|
|
|
items.length
|
|
|
|
} items, including .textileseed`
|
|
|
|
);
|
|
|
|
}
|
2020-10-06 22:27:27 +03:00
|
|
|
}
|
|
|
|
|
2020-10-06 22:07:43 +03:00
|
|
|
// NOTE(jim): Remove the bucket from Textile.
|
2020-10-06 23:35:22 +03:00
|
|
|
// ----------------------------------------------------------------
|
2020-10-06 22:07:43 +03:00
|
|
|
let bucketRemoval;
|
|
|
|
try {
|
|
|
|
bucketRemoval = await buckets.remove(list[0].key);
|
|
|
|
} catch (e) {
|
2020-10-06 23:35:22 +03:00
|
|
|
reportError(e.message);
|
2020-10-06 22:07:43 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE(jim): Verify we cleaned up user data.
|
2020-10-06 23:35:22 +03:00
|
|
|
// ----------------------------------------------------------------
|
|
|
|
list = "CHANGE_ME";
|
2020-10-06 22:07:43 +03:00
|
|
|
try {
|
|
|
|
list = await buckets.list();
|
|
|
|
console.log({ emptyList: list });
|
|
|
|
} catch (e) {
|
2020-10-06 23:35:22 +03:00
|
|
|
reportError(e.message);
|
2020-10-06 22:07:43 +03:00
|
|
|
}
|
2020-10-06 23:35:22 +03:00
|
|
|
|
|
|
|
reportTask(`SUCCESS.`);
|
2020-10-06 22:07:43 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
run();
|
|
|
|
|
|
|
|
console.log(`FINISHED: recreate-bucket-bug.js`);
|
2020-10-07 00:47:16 +03:00
|
|
|
console.log(` CTRL +C to return to terminal`);
|