From 82d4ff98f1a0ac2494713d39543edfde8c8c5dcf Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Wed, 25 Jun 2025 22:24:34 -0400 Subject: [PATCH 01/11] feat(migrate): init NPM package --- utils/migrate/package.json | 12 ++++++++++++ 1 file changed, 12 insertions(+) create mode 100644 utils/migrate/package.json diff --git a/utils/migrate/package.json b/utils/migrate/package.json new file mode 100644 index 0000000..78a6ad2 --- /dev/null +++ b/utils/migrate/package.json @@ -0,0 +1,12 @@ +{ + "name": "ruby-ghost-migrator", + "version": "0.1.0", + "description": "", + "main": "index.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "keywords": [], + "author": "Ruby Central", + "license": "private" +} From 466848ee6cc0dd14fd8869d7f375fc66339b0424 Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Wed, 25 Jun 2025 22:26:28 -0400 Subject: [PATCH 02/11] feat(migrate): add dotenv package * Ignore node_modules, .env, and DS_Store --- utils/migrate/.gitignore | 3 +++ utils/migrate/package-lock.json | 27 +++++++++++++++++++++++++++ utils/migrate/package.json | 5 ++++- 3 files changed, 34 insertions(+), 1 deletion(-) create mode 100644 utils/migrate/.gitignore create mode 100644 utils/migrate/package-lock.json diff --git a/utils/migrate/.gitignore b/utils/migrate/.gitignore new file mode 100644 index 0000000..0d777ad --- /dev/null +++ b/utils/migrate/.gitignore @@ -0,0 +1,3 @@ +node_modules/ +*.DS_Store +.env diff --git a/utils/migrate/package-lock.json b/utils/migrate/package-lock.json new file mode 100644 index 0000000..21d35ee --- /dev/null +++ b/utils/migrate/package-lock.json @@ -0,0 +1,27 @@ +{ + "name": "ruby-ghost-migrator", + "version": "0.1.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "ruby-ghost-migrator", + "version": "0.1.0", + "license": "private", + "dependencies": { + "dotenv": "^16.5.0" + } + }, + "node_modules/dotenv": { + "version": "16.5.0", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.5.0.tgz", + "integrity": "sha512-m/C+AwOAr9/W1UOIZUo232ejMNnJAJtYQjUbHoNTBNTJSvqzzDh7vnrei3o3r3m9blf6ZoDkvcw0VmozNRFJxg==", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://dotenvx.com" + } + } + } +} diff --git a/utils/migrate/package.json b/utils/migrate/package.json index 78a6ad2..21b36b1 100644 --- a/utils/migrate/package.json +++ b/utils/migrate/package.json @@ -8,5 +8,8 @@ }, "keywords": [], "author": "Ruby Central", - "license": "private" + "license": "private", + "dependencies": { + "dotenv": "^16.5.0" + } } From 73ef6557b6a0422ff420a3a110291d1abe5e1f0b Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Wed, 25 Jun 2025 22:31:16 -0400 Subject: [PATCH 03/11] feat(migrate): add ghost api client --- utils/migrate/package-lock.json | 390 ++++++++++++++++++++++++++++++++ utils/migrate/package.json | 1 + 2 files changed, 391 insertions(+) diff --git a/utils/migrate/package-lock.json b/utils/migrate/package-lock.json index 21d35ee..6a382a7 100644 --- a/utils/migrate/package-lock.json +++ b/utils/migrate/package-lock.json @@ -9,9 +9,71 @@ "version": "0.1.0", "license": "private", "dependencies": { + "@tryghost/admin-api": "^1.13.17", "dotenv": "^16.5.0" } }, + "node_modules/@tryghost/admin-api": { + "version": "1.13.17", + "resolved": "https://registry.npmjs.org/@tryghost/admin-api/-/admin-api-1.13.17.tgz", + "integrity": "sha512-FTOI5JxXcOllc66CNO/uByO0CTynY3A9qUqQSI3FcOmfPDtAU6i1epoMO0R8kjo0lsQIb3S6LRnbBdgduUaoDQ==", + "dependencies": { + "axios": "^1.0.0", + "form-data": "^4.0.0", + "jsonwebtoken": "^9.0.0" + } + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" + }, + "node_modules/axios": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.10.0.tgz", + "integrity": "sha512-/1xYAC4MP/HEG+3duIhFr4ZQXR4sQXOIe+o6sdqzeykGLx6Upp/1p8MHqhINOvGeP7xyNHe7tsiJByc4SSVUxw==", + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==" + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", + "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/dotenv": { "version": "16.5.0", "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.5.0.tgz", @@ -22,6 +84,334 @@ "funding": { "url": "https://dotenvx.com" } + }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "dependencies": { + "safe-buffer": "^5.0.1" + } + }, + "node_modules/es-define-property": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-object-atoms": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", + "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-set-tostringtag": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", + "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", + "dependencies": { + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.6", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/follow-redirects": { + "version": "1.15.9", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", + "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/form-data": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.3.tgz", + "integrity": "sha512-qsITQPfmvMOSAdeyZ+12I1c+CKSstAFAwu+97zrnWAbIr5u8wfsExUzCesVLC8NgHuRUqNN4Zy6UPWUTRGslcA==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "es-set-tostringtag": "^2.1.0", + "hasown": "^2.0.2", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-intrinsic": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "function-bind": "^1.1.2", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", + "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-tostringtag": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", + "dependencies": { + "has-symbols": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/jsonwebtoken": { + "version": "9.0.2", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz", + "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==", + "dependencies": { + "jws": "^3.2.2", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=12", + "npm": ">=6" + } + }, + "node_modules/jwa": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.2.tgz", + "integrity": "sha512-eeH5JO+21J78qMvTIDdBXidBd6nG2kZjg5Ohz/1fpa28Z4CcsWUzJ1ZZyFq/3z3N17aZy+ZuBoHljASbL1WfOw==", + "dependencies": { + "buffer-equal-constant-time": "^1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", + "dependencies": { + "jwa": "^1.4.1", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==" + }, + "node_modules/lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==" + }, + "node_modules/lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==" + }, + "node_modules/lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==" + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==" + }, + "node_modules/lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==" + }, + "node_modules/lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==" + }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" + }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/semver": { + "version": "7.7.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.2.tgz", + "integrity": "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } } } } diff --git a/utils/migrate/package.json b/utils/migrate/package.json index 21b36b1..42ba036 100644 --- a/utils/migrate/package.json +++ b/utils/migrate/package.json @@ -10,6 +10,7 @@ "author": "Ruby Central", "license": "private", "dependencies": { + "@tryghost/admin-api": "^1.13.17", "dotenv": "^16.5.0" } } From 7f2b9f8eff8341edc2a5e00b31d073cdbba16419 Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 01:08:26 -0400 Subject: [PATCH 04/11] feat(migrate): create function for temp files from page payloads * Set up settings json file * Create remove temp files function --- utils/migrate/config/settings.json | 24 ++++++++++++ utils/migrate/index.js | 61 ++++++++++++++++++++++++++++++ utils/migrate/package.json | 1 + utils/migrate/tmp/.keep | 0 4 files changed, 86 insertions(+) create mode 100644 utils/migrate/config/settings.json create mode 100644 utils/migrate/index.js create mode 100644 utils/migrate/tmp/.keep diff --git a/utils/migrate/config/settings.json b/utils/migrate/config/settings.json new file mode 100644 index 0000000..301454e --- /dev/null +++ b/utils/migrate/config/settings.json @@ -0,0 +1,24 @@ +{ + "slugs": [ + "about-hero-body", + "about-hero-button", + "about-history-sectionone-body", + "about-history-sectionone-button", + "about-history-sectionthree-body", + "about-history-sectiontwo-body", + "about-position-body", + "about-team-directors-header", + "about-team-header", + "about-team-staff-header", + "footer-socials", + "home-community", + "home-hero-body", + "home-hero-button-left", + "home-hero-button-right", + "home-opensource", + "home-opensource-button", + "home-pillars-button", + "home-pillars-conferences", + "home-support" + ] +} diff --git a/utils/migrate/index.js b/utils/migrate/index.js new file mode 100644 index 0000000..613333a --- /dev/null +++ b/utils/migrate/index.js @@ -0,0 +1,61 @@ +/* eslint-disable no-console */ + +import { readFileSync, rmSync, writeFileSync } from 'fs'; +import { join } from 'path'; +import 'dotenv/config'; +import GhostAdminAPI from '@tryghost/admin-api'; + +const STAGING = 'staging'; +const PRODUCTION = 'production'; + +const clientStaging = new GhostAdminAPI({ + url: process.env.STAGING_API_URL, + key: process.env.STAGING_API_KEY_ADMIN, + version: 'v5.126.0', +}); + +const clientProduction = new GhostAdminAPI({ + url: process.env.PRODUCTION_API_URL, + key: process.env.PRODUCTION_API_KEY_ADMIN, + version: 'v5.126.0', +}); + +const getPages = (target) => { + const normalizedTarget = target.toLowerCase(); + const client = + normalizedTarget === STAGING ? clientStaging : clientProduction; + const dataPath = join(process.cwd(), `tmp/${normalizedTarget}.json`); + const settingsPath = join(process.cwd(), 'config/settings.json'); + const settings = JSON.parse(readFileSync(settingsPath)); + const targetSlugs = settings.slugs.join(','); + + console.log(`Fetching data from ${normalizedTarget} ...\n`); + + return new Promise((resolve, reject) => { + client.pages + .browse({ filter: `slug:[${targetSlugs}]` }) + .then((pages) => { + const jsonString = `{ "pages": ${JSON.stringify(pages)} }`; + writeFileSync(dataPath, jsonString, { flag: 'a' }); + resolve(); + }) + .catch((e) => reject(e)); + }); +}; + +const cleanTmpFile = (target) => { + const relativePath = `tmp/${target.toLowerCase()}.json`; + + console.log(`Removing "${relativePath}"...\n`); + rmSync(join(process.cwd(), relativePath)); +}; + +getPages(STAGING).then(() => { + getPages(PRODUCTION) + .then(() => { + cleanTmpFile(STAGING); + }) + .then(() => { + cleanTmpFile(PRODUCTION); + }); +}); diff --git a/utils/migrate/package.json b/utils/migrate/package.json index 42ba036..50dac3e 100644 --- a/utils/migrate/package.json +++ b/utils/migrate/package.json @@ -9,6 +9,7 @@ "keywords": [], "author": "Ruby Central", "license": "private", + "type": "module", "dependencies": { "@tryghost/admin-api": "^1.13.17", "dotenv": "^16.5.0" diff --git a/utils/migrate/tmp/.keep b/utils/migrate/tmp/.keep new file mode 100644 index 0000000..e69de29 From 4a74464e8841f91695178deb2290b6abffd8ae0d Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 03:05:03 -0400 Subject: [PATCH 05/11] feat(migrate): ignore everything except .keep in tmp directory --- utils/migrate/.gitignore | 2 ++ 1 file changed, 2 insertions(+) diff --git a/utils/migrate/.gitignore b/utils/migrate/.gitignore index 0d777ad..7502f98 100644 --- a/utils/migrate/.gitignore +++ b/utils/migrate/.gitignore @@ -1,3 +1,5 @@ node_modules/ *.DS_Store .env +tmp/* +!tmp/.keep From 11f6c206ea5aa1dac67aa99ea0b3816d235f5a0c Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 03:07:05 -0400 Subject: [PATCH 06/11] feat(migrate): create page copies on production --- utils/migrate/index.js | 35 +++++++++++++++++++++++++---------- 1 file changed, 25 insertions(+), 10 deletions(-) diff --git a/utils/migrate/index.js b/utils/migrate/index.js index 613333a..4814f3e 100644 --- a/utils/migrate/index.js +++ b/utils/migrate/index.js @@ -29,11 +29,11 @@ const getPages = (target) => { const settings = JSON.parse(readFileSync(settingsPath)); const targetSlugs = settings.slugs.join(','); - console.log(`Fetching data from ${normalizedTarget} ...\n`); + console.log(`Fetching data from ${normalizedTarget} ...\n\n`); return new Promise((resolve, reject) => { client.pages - .browse({ filter: `slug:[${targetSlugs}]` }) + .browse({ filter: `slug:[${targetSlugs}]`, limit: 1000 }) .then((pages) => { const jsonString = `{ "pages": ${JSON.stringify(pages)} }`; writeFileSync(dataPath, jsonString, { flag: 'a' }); @@ -46,16 +46,31 @@ const getPages = (target) => { const cleanTmpFile = (target) => { const relativePath = `tmp/${target.toLowerCase()}.json`; - console.log(`Removing "${relativePath}"...\n`); + console.log(`Removing "${relativePath}"...\n\n`); rmSync(join(process.cwd(), relativePath)); }; -getPages(STAGING).then(() => { - getPages(PRODUCTION) - .then(() => { - cleanTmpFile(STAGING); - }) - .then(() => { - cleanTmpFile(PRODUCTION); +const moveToProduction = () => { + const stagingDataPath = join(process.cwd(), 'tmp/staging.json'); + const stagingData = JSON.parse(readFileSync(stagingDataPath)).pages; + + console.log(`Adding page copies from ${STAGING} to ${PRODUCTION}...\n\n`); + + return new Promise((resolve) => { + stagingData.forEach((stagingPage) => { + clientProduction.pages.add({ + lexical: stagingPage.lexical, + status: 'published', + title: stagingPage.title, + published_at: stagingPage.published_at, + }); + resolve(); }); + }); +}; + +getPages(STAGING).then(() => { + moveToProduction().then(() => { + cleanTmpFile(STAGING); + }); }); From 01eb37aa15737ee7d113c06e7764d1d114c505e1 Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 03:10:49 -0400 Subject: [PATCH 07/11] feat(migrate): add example env --- utils/migrate/.env.example | 7 +++++++ utils/migrate/.gitignore | 3 ++- 2 files changed, 9 insertions(+), 1 deletion(-) create mode 100644 utils/migrate/.env.example diff --git a/utils/migrate/.env.example b/utils/migrate/.env.example new file mode 100644 index 0000000..13ae7e2 --- /dev/null +++ b/utils/migrate/.env.example @@ -0,0 +1,7 @@ +STAGING_API_KEY_CONTENT= +STAGING_API_KEY_ADMIN= +STAGING_API_URL= + +PRODUCTION_API_KEY_CONTENT= +PRODUCTION_API_KEY_ADMIN= +PRODUCTION_API_URL= diff --git a/utils/migrate/.gitignore b/utils/migrate/.gitignore index 7502f98..af1dae5 100644 --- a/utils/migrate/.gitignore +++ b/utils/migrate/.gitignore @@ -1,5 +1,6 @@ node_modules/ *.DS_Store -.env tmp/* !tmp/.keep +.env* +!.env.example From 77c682098cb11120a1a9feabfae6fafc2c2bbd33 Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 03:14:00 -0400 Subject: [PATCH 08/11] refactor(settings.json): run Prettier --- utils/migrate/config/settings.json | 44 +++++++++++++++--------------- 1 file changed, 22 insertions(+), 22 deletions(-) diff --git a/utils/migrate/config/settings.json b/utils/migrate/config/settings.json index 301454e..59d3f3d 100644 --- a/utils/migrate/config/settings.json +++ b/utils/migrate/config/settings.json @@ -1,24 +1,24 @@ { - "slugs": [ - "about-hero-body", - "about-hero-button", - "about-history-sectionone-body", - "about-history-sectionone-button", - "about-history-sectionthree-body", - "about-history-sectiontwo-body", - "about-position-body", - "about-team-directors-header", - "about-team-header", - "about-team-staff-header", - "footer-socials", - "home-community", - "home-hero-body", - "home-hero-button-left", - "home-hero-button-right", - "home-opensource", - "home-opensource-button", - "home-pillars-button", - "home-pillars-conferences", - "home-support" - ] + "slugs": [ + "about-hero-body", + "about-hero-button", + "about-history-sectionone-body", + "about-history-sectionone-button", + "about-history-sectionthree-body", + "about-history-sectiontwo-body", + "about-position-body", + "about-team-directors-header", + "about-team-header", + "about-team-staff-header", + "footer-socials", + "home-community", + "home-hero-body", + "home-hero-button-left", + "home-hero-button-right", + "home-opensource", + "home-opensource-button", + "home-pillars-button", + "home-pillars-conferences", + "home-support" + ] } From a3ede6831e2a98daed893d19edf2111b1ee4ca54 Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 03:14:25 -0400 Subject: [PATCH 09/11] feat(migrate): add migrate alias --- utils/migrate/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/utils/migrate/package.json b/utils/migrate/package.json index 50dac3e..d176681 100644 --- a/utils/migrate/package.json +++ b/utils/migrate/package.json @@ -4,7 +4,7 @@ "description": "", "main": "index.js", "scripts": { - "test": "echo \"Error: no test specified\" && exit 1" + "migrate": "echo \"Running migration from staging to production...\n\n\" && node index.js" }, "keywords": [], "author": "Ruby Central", From 04d197ac7e7d7572badb90dac6191c038d249ea6 Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 03:24:55 -0400 Subject: [PATCH 10/11] fix(migrate): update package name --- utils/migrate/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/utils/migrate/package.json b/utils/migrate/package.json index d176681..42f05e9 100644 --- a/utils/migrate/package.json +++ b/utils/migrate/package.json @@ -1,5 +1,5 @@ { - "name": "ruby-ghost-migrator", + "name": "ruby-central-ghost-migrator", "version": "0.1.0", "description": "", "main": "index.js", From e470fe4f1f20d1d2471dcc80ad5a2d44f15ba4ee Mon Sep 17 00:00:00 2001 From: Sunjay Armstead <65554107+sarmstead@users.noreply.github.com> Date: Thu, 26 Jun 2025 03:25:14 -0400 Subject: [PATCH 11/11] feat(migrate): add README --- utils/migrate/README.md | 12 ++++++++++++ 1 file changed, 12 insertions(+) create mode 100644 utils/migrate/README.md diff --git a/utils/migrate/README.md b/utils/migrate/README.md new file mode 100644 index 0000000..73b54c4 --- /dev/null +++ b/utils/migrate/README.md @@ -0,0 +1,12 @@ +# Ruby Central Ghost Migrator + +The purpose of this package is to copy data from a staging Ghost environment to a production Ghost environment. + +## Setup + +1. The migration script reads the `slugs` key from `./config/settings.json` to round up a list of posts to port over to production. Make sure that array is up to date for your needs. +1. Create a `.env` file using the `.env.example` file as a guide. You will need to follow [Ghost's integration instructions](https://ghost.org/docs/admin-api/) to retrieve your API keys. + +## How to run a migration + +In this directory, run the command `npm run migrate`. Then verify that the production site has the pages you expect.