From 0db04cf5cdea93ba70ee67c772f1dbfdfff5516a Mon Sep 17 00:00:00 2001 From: claude-bot Date: Fri, 15 May 2026 16:42:00 +0900 Subject: [PATCH] feat: implement video site per README spec MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Express + EJS + express-session stack (auth/navbar ported from minecraft_launcher) - Public: main folder list, folder video grid, internal popup player (/player/:videoId) - Admin (/op): login, folder CRUD with right-click context menu + add-folder modal - Admin folder: video grid with right-click edit/rename/delete, "영상 추가" -> editor - Video editor: drag-drop upload, file picker, YouTube URL probe (ETA + 5분 경고), background yt-dlp download with progress polling, navbar title edit, trim controls, save runs ffmpeg trim (original preserved) - Filesystem storage under data/folders///{meta.json, original., edited.} Co-Authored-By: Claude Opus 4.7 --- .gitignore | 11 + README.md | 30 + account.json | 6 + data/folders/.gitkeep | 0 data/jobs/.gitkeep | 0 data/tmp/.gitkeep | 0 package-lock.json | 1193 +++++++++++++++++++++++++++++++++++++ package.json | 26 + public/dashboard.js | 80 +++ public/editor.js | 197 ++++++ public/folder.js | 59 ++ public/player.js | 51 ++ public/styles.css | 234 ++++++++ src/app.ts | 70 +++ src/auth.ts | 19 + src/editor.ts | 93 +++ src/paths.ts | 19 + src/routes/op.ts | 300 ++++++++++ src/routes/public.ts | 94 +++ src/store.ts | 220 +++++++ src/youtube.ts | 243 ++++++++ tsconfig.json | 15 + views/folder.ejs | 50 ++ views/index.ejs | 37 ++ views/op/dashboard.ejs | 56 ++ views/op/editor.ejs | 64 ++ views/op/folder.ejs | 50 ++ views/op/login.ejs | 24 + views/partials/navbar.ejs | 30 + views/player.ejs | 29 + 30 files changed, 3300 insertions(+) create mode 100644 .gitignore create mode 100644 account.json create mode 100644 data/folders/.gitkeep create mode 100644 data/jobs/.gitkeep create mode 100644 data/tmp/.gitkeep create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 public/dashboard.js create mode 100644 public/editor.js create mode 100644 public/folder.js create mode 100644 public/player.js create mode 100644 public/styles.css create mode 100644 src/app.ts create mode 100644 src/auth.ts create mode 100644 src/editor.ts create mode 100644 src/paths.ts create mode 100644 src/routes/op.ts create mode 100644 src/routes/public.ts create mode 100644 src/store.ts create mode 100644 src/youtube.ts create mode 100644 tsconfig.json create mode 100644 views/folder.ejs create mode 100644 views/index.ejs create mode 100644 views/op/dashboard.ejs create mode 100644 views/op/editor.ejs create mode 100644 views/op/folder.ejs create mode 100644 views/op/login.ejs create mode 100644 views/partials/navbar.ejs create mode 100644 views/player.ejs diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..39f148d --- /dev/null +++ b/.gitignore @@ -0,0 +1,11 @@ +node_modules/ +dist/ +data/folders/* +!data/folders/.gitkeep +data/jobs/* +!data/jobs/.gitkeep +data/tmp/* +!data/tmp/.gitkeep +.env +*.log +.DS_Store diff --git a/README.md b/README.md index 76962b8..cbd8fd2 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,36 @@ # 비디오 사이트 ## 영상 업로드 및 저장을 위한 사이트 +## 실행 + +```bash +npm install +npm run build +npm start # 기본 http://127.0.0.1:3000 (PORT=3000, HOST=127.0.0.1) +``` + +- 외부 노출이 필요하면 `HOST=0.0.0.0 npm start` +- 관리자 비밀번호는 `account.json` 의 `password` 값 (초기값 `admin`, 운영 시 반드시 변경) +- 세션 비밀은 `SESSION_SECRET` 환경변수로 덮어쓰기 권장 + +## 외부 의존 + +- `yt-dlp` — YouTube 영상 가져오기 (`PATH` 또는 `./bin/yt-dlp` 에 설치) +- `ffmpeg` — 영상 트림 저장 (`PATH` 에 설치). 없으면 trim 설정만 저장됩니다. + +## 데이터 위치 + +``` +data/ + folders/<폴더이름>// + meta.json # 영상 메타 (제목, trim, 원본/편집본 파일명) + original. # 원본 (항상 보존) + edited. # 편집본 (저장 시 생성) + jobs/.json # YouTube 다운로드 작업 상태 +``` + +## 스펙 + ### 메인 페이지 (/) - 동영상이 저장되어있는 폴더를 나열합니다. - 폴더를 선택해 안에 영상을 확인할수있습니다. diff --git a/account.json b/account.json new file mode 100644 index 0000000..adc6e63 --- /dev/null +++ b/account.json @@ -0,0 +1,6 @@ +[ + { + "id": "admin", + "password": "admin" + } +] diff --git a/data/folders/.gitkeep b/data/folders/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/data/jobs/.gitkeep b/data/jobs/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/data/tmp/.gitkeep b/data/tmp/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..5e8f9f6 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,1193 @@ +{ + "name": "make-video-site", + "version": "0.1.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "make-video-site", + "version": "0.1.0", + "dependencies": { + "ejs": "^3.1.10", + "express": "^4.19.2", + "express-session": "^1.18.0", + "multer": "^1.4.5-lts.1" + }, + "devDependencies": { + "@types/ejs": "^3.1.5", + "@types/express": "^4.17.21", + "@types/express-session": "^1.18.0", + "@types/multer": "^1.4.11", + "@types/node": "^22.5.0", + "typescript": "^5.5.4" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.6", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.6.tgz", + "integrity": "sha512-HLFeCYgz89uk22N5Qg3dvGvsv46B8GLvKKo1zKG4NybA8U2DiEO3w9lqGg29t/tfLRJpJ6iQxnVw4OnB7MoM9g==", + "dev": true, + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.38", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", + "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/ejs": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/@types/ejs/-/ejs-3.1.5.tgz", + "integrity": "sha512-nv+GSx77ZtXiJzwKdsASqi+YQ5Z7vwHsTP0JY2SiQgjGckkBRKZnk8nIM+7oUZ1VCtuTz0+By4qVR7fqzp/Dfg==", + "dev": true + }, + "node_modules/@types/express": { + "version": "4.17.25", + "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.25.tgz", + "integrity": "sha512-dVd04UKsfpINUnK0yBoYHDF3xu7xVH4BuDotC/xGuycx4CgbP48X/KF/586bcObxT0HENHXEU8Nqtu6NR+eKhw==", + "dev": true, + "dependencies": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^4.17.33", + "@types/qs": "*", + "@types/serve-static": "^1" + } + }, + "node_modules/@types/express-serve-static-core": { + "version": "4.19.8", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.19.8.tgz", + "integrity": "sha512-02S5fmqeoKzVZCHPZid4b8JH2eM5HzQLZWN2FohQEy/0eXTq8VXZfSN6Pcr3F6N9R/vNrj7cpgbhjie6m/1tCA==", + "dev": true, + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*", + "@types/send": "*" + } + }, + "node_modules/@types/express-session": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/@types/express-session/-/express-session-1.19.0.tgz", + "integrity": "sha512-GbypG0bog68UbOq2tSAp7SclvCUm3ha1uDi58OPRGK1NfRvCIu7Gz0M7fTGtpNG1T9a29GpuurQj9zEcT/lMXQ==", + "dev": true, + "dependencies": { + "@types/express": "*" + } + }, + "node_modules/@types/http-errors": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.5.tgz", + "integrity": "sha512-r8Tayk8HJnX0FztbZN7oVqGccWgw98T/0neJphO91KkmOzug1KkofZURD4UaD5uH8AqcFLfdPErnBod0u71/qg==", + "dev": true + }, + "node_modules/@types/mime": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz", + "integrity": "sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==", + "dev": true + }, + "node_modules/@types/multer": { + "version": "1.4.13", + "resolved": "https://registry.npmjs.org/@types/multer/-/multer-1.4.13.tgz", + "integrity": "sha512-bhhdtPw7JqCiEfC9Jimx5LqX9BDIPJEh2q/fQ4bqbBPtyEZYr3cvF22NwG0DmPZNYA0CAf2CnqDB4KIGGpJcaw==", + "dev": true, + "dependencies": { + "@types/express": "*" + } + }, + "node_modules/@types/node": { + "version": "22.19.19", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.19.19.tgz", + "integrity": "sha512-dyh/xO2Fh5bYrfWaaqGrRQQGkNdmYw6AmaAUvYeUMNTWQtvb796ikLdmTchRmOlOiIJ1TDXfWgVx1QkUlQ6Hew==", + "dev": true, + "dependencies": { + "undici-types": "~6.21.0" + } + }, + "node_modules/@types/qs": { + "version": "6.15.1", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.15.1.tgz", + "integrity": "sha512-GZHUBZR9hckSUhrxmp1nG6NwdpM9fCunJwyThLW1X3AyHgd9IlHb6VANpQQqDr2o/qQp6McZ3y/IA2rVzKzSbw==", + "dev": true + }, + "node_modules/@types/range-parser": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz", + "integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==", + "dev": true + }, + "node_modules/@types/send": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@types/send/-/send-1.2.1.tgz", + "integrity": "sha512-arsCikDvlU99zl1g69TcAB3mzZPpxgw0UQnaHeC1Nwb015xp8bknZv5rIfri9xTOcMuaVgvabfIRA7PSZVuZIQ==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/serve-static": { + "version": "1.15.10", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.10.tgz", + "integrity": "sha512-tRs1dB+g8Itk72rlSI2ZrW6vZg0YrLI81iQSTkMmOqnqCaNr/8Ek4VwWcN5vZgCYWbg/JJSGBlUaYGAOP73qBw==", + "dev": true, + "dependencies": { + "@types/http-errors": "*", + "@types/node": "*", + "@types/send": "<1" + } + }, + "node_modules/@types/serve-static/node_modules/@types/send": { + "version": "0.17.6", + "resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.6.tgz", + "integrity": "sha512-Uqt8rPBE8SY0RK8JB1EzVOIZ32uqy8HwdxCnoCOsYrvnswqmFZ/k+9Ikidlk/ImhsdvBsloHbAlewb2IEBV/Og==", + "dev": true, + "dependencies": { + "@types/mime": "^1", + "@types/node": "*" + } + }, + "node_modules/accepts": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", + "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", + "dependencies": { + "mime-types": "~2.1.34", + "negotiator": "0.6.3" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/append-field": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz", + "integrity": "sha512-klpgFSWLW1ZEs8svjfb7g4qWY0YS5imI82dTg+QahUvJ8YqAY0P10Uk8tTyh9ZGuYEZEMaeJYCF5BFuX552hsw==" + }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" + }, + "node_modules/async": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", + "integrity": "sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==" + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" + }, + "node_modules/body-parser": { + "version": "1.20.5", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.5.tgz", + "integrity": "sha512-3grm+/2tUOvu2cjJkvsIxrv/wVpfXQW4PsQHYm7yk4vfpu7Ekl6nEsYBoJUL6qDwZUx8wUhQ8tR2qz+ad9c9OA==", + "dependencies": { + "bytes": "~3.1.2", + "content-type": "~1.0.5", + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "~1.2.0", + "http-errors": "~2.0.1", + "iconv-lite": "~0.4.24", + "on-finished": "~2.4.1", + "qs": "~6.15.1", + "raw-body": "~2.5.3", + "type-is": "~1.6.18", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/brace-expansion": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.1.0.tgz", + "integrity": "sha512-TN1kCZAgdgweJhWWpgKYrQaMNHcDULHkWwQIspdtjV4Y5aurRdZpjAqn6yX3FPqTA9ngHCc4hJxMAMgGfve85w==", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" + }, + "node_modules/busboy": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz", + "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==", + "dependencies": { + "streamsearch": "^1.1.0" + }, + "engines": { + "node": ">=10.16.0" + } + }, + "node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "engines": { + "node": ">= 0.8" + } + }, + "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/call-bound": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", + "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "get-intrinsic": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/concat-stream": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", + "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", + "engines": [ + "node >= 0.8" + ], + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^2.2.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/content-disposition": { + "version": "0.5.4", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", + "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", + "dependencies": { + "safe-buffer": "5.2.1" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", + "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz", + "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.7.tgz", + "integrity": "sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==" + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==" + }, + "node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/destroy": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", + "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "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/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" + }, + "node_modules/ejs": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.10.tgz", + "integrity": "sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA==", + "dependencies": { + "jake": "^10.8.5" + }, + "bin": { + "ejs": "bin/cli.js" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/encodeurl": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", + "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", + "engines": { + "node": ">= 0.8" + } + }, + "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/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express": { + "version": "4.22.2", + "resolved": "https://registry.npmjs.org/express/-/express-4.22.2.tgz", + "integrity": "sha512-IuL+Elrou2ZvCFHs18/CIzy2Nzvo25nZ1/D2eIZlz7c+QUayAcYoiM2BthCjs+EBHVpjYjcuLDAiCWgeIX3X1Q==", + "dependencies": { + "accepts": "~1.3.8", + "array-flatten": "1.1.1", + "body-parser": "~1.20.5", + "content-disposition": "~0.5.4", + "content-type": "~1.0.4", + "cookie": "~0.7.1", + "cookie-signature": "~1.0.6", + "debug": "2.6.9", + "depd": "2.0.0", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.3.1", + "fresh": "~0.5.2", + "http-errors": "~2.0.0", + "merge-descriptors": "1.0.3", + "methods": "~1.1.2", + "on-finished": "~2.4.1", + "parseurl": "~1.3.3", + "path-to-regexp": "~0.1.12", + "proxy-addr": "~2.0.7", + "qs": "~6.15.1", + "range-parser": "~1.2.1", + "safe-buffer": "5.2.1", + "send": "~0.19.0", + "serve-static": "~1.16.2", + "setprototypeof": "1.2.0", + "statuses": "~2.0.1", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/express-session": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/express-session/-/express-session-1.19.0.tgz", + "integrity": "sha512-0csaMkGq+vaiZTmSMMGkfdCOabYv192VbytFypcvI0MANrp+4i/7yEkJ0sbAEhycQjntaKGzYfjfXQyVb7BHMA==", + "dependencies": { + "cookie": "~0.7.2", + "cookie-signature": "~1.0.7", + "debug": "~2.6.9", + "depd": "~2.0.0", + "on-headers": "~1.1.0", + "parseurl": "~1.3.3", + "safe-buffer": "~5.2.1", + "uid-safe": "~2.1.5" + }, + "engines": { + "node": ">= 0.8.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/filelist": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.6.tgz", + "integrity": "sha512-5giy2PkLYY1cP39p17Ech+2xlpTRL9HLspOfEgm0L6CwBXBTgsK5ou0JtzYuepxkaQ/tvhCFIJ5uXo0OrM2DxA==", + "dependencies": { + "minimatch": "^5.0.1" + } + }, + "node_modules/finalhandler": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.2.tgz", + "integrity": "sha512-aA4RyPcd3badbdABGDuTXCMTtOneUCAYH/gxoYRTZlIJdF0YPWuGqiAsIrhNnnqdXGswYk6dGujem4w80UJFhg==", + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "on-finished": "~2.4.1", + "parseurl": "~1.3.3", + "statuses": "~2.0.2", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", + "engines": { + "node": ">= 0.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/hasown": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.3.tgz", + "integrity": "sha512-ej4AhfhfL2Q2zpMmLo7U1Uv9+PyhIZpgQLGT1F9miIGmiCJIoCgSmczFdrc97mWT4kVY72KA+WnnhJ5pghSvSg==", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/http-errors": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz", + "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==", + "dependencies": { + "depd": "~2.0.0", + "inherits": "~2.0.4", + "setprototypeof": "~1.2.0", + "statuses": "~2.0.2", + "toidentifier": "~1.0.1" + }, + "engines": { + "node": ">= 0.8" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==" + }, + "node_modules/jake": { + "version": "10.9.4", + "resolved": "https://registry.npmjs.org/jake/-/jake-10.9.4.tgz", + "integrity": "sha512-wpHYzhxiVQL+IV05BLE2Xn34zW1S223hvjtqk0+gsPrwd/8JNLXJgZZM/iPFsYc1xyphF+6M6EvdE5E9MBGkDA==", + "dependencies": { + "async": "^3.2.6", + "filelist": "^1.0.4", + "picocolors": "^1.1.1" + }, + "bin": { + "jake": "bin/cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "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/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", + "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=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/minimatch": { + "version": "5.1.9", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.9.tgz", + "integrity": "sha512-7o1wEA2RyMP7Iu7GNba9vc0RWWGACJOCZBJX2GJWip0ikV+wcOsgVuY9uE8CPiyQhkGFSlhuSkZPavN7u1c2Fw==", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/multer": { + "version": "1.4.5-lts.2", + "resolved": "https://registry.npmjs.org/multer/-/multer-1.4.5-lts.2.tgz", + "integrity": "sha512-VzGiVigcG9zUAoCNU+xShztrlr1auZOlurXynNvO9GiWD1/mTBbUljOKY+qMeazBqXgRnjzeEgJI/wyjJUHg9A==", + "deprecated": "Multer 1.x is impacted by a number of vulnerabilities, which have been patched in 2.x. You should upgrade to the latest 2.x version.", + "dependencies": { + "append-field": "^1.0.0", + "busboy": "^1.0.0", + "concat-stream": "^1.5.2", + "mkdirp": "^0.5.4", + "object-assign": "^4.1.1", + "type-is": "^1.6.4", + "xtend": "^4.0.0" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { + "version": "1.13.4", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", + "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/on-headers": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.1.0.tgz", + "integrity": "sha512-737ZY3yNnXy37FHkQxPzt4UZ2UWPWiCZWLvFZ4fu5cueciegX0zGPnrlY6bwRg4FdQOe9YU8MkmJwGhoMybl8A==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/path-to-regexp": { + "version": "0.1.13", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.13.tgz", + "integrity": "sha512-A/AGNMFN3c8bOlvV9RreMdrv7jsmF9XIfDeCd87+I8RNg6s78BhJxMu69NEMHBSJFxKidViTEdruRwEk/WIKqA==" + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==" + }, + "node_modules/process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/qs": { + "version": "6.15.1", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.15.1.tgz", + "integrity": "sha512-6YHEFRL9mfgcAvql/XhwTvf5jKcOiiupt2FiJxHkiX1z4j7WL8J/jRHYLluORvc1XxB5rV20KoeK00gVJamspg==", + "dependencies": { + "side-channel": "^1.1.0" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/random-bytes": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/random-bytes/-/random-bytes-1.0.0.tgz", + "integrity": "sha512-iv7LhNVO047HzYR3InF6pUcUsPQiHTM1Qal51DcGSuZFBil1aBBWG5eHPNek7bvILMaYJ/8RU1e8w1AMdHmLQQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.5.3", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.3.tgz", + "integrity": "sha512-s4VSOf6yN0rvbRZGxs8Om5CWj6seneMwK3oDb4lWDH0UPhWcxwOWw5+qk24bxq87szX1ydrwylIOp2uG1ojUpA==", + "dependencies": { + "bytes": "~3.1.2", + "http-errors": "~2.0.1", + "iconv-lite": "~0.4.24", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/readable-stream/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "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/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/send": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/send/-/send-0.19.2.tgz", + "integrity": "sha512-VMbMxbDeehAxpOtWJXlcUS5E8iXh6QmN+BkRX1GARS3wRaXEEgzCcB10gTQazO42tpNIya8xIyNx8fll1OFPrg==", + "dependencies": { + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "~0.5.2", + "http-errors": "~2.0.1", + "mime": "1.6.0", + "ms": "2.1.3", + "on-finished": "~2.4.1", + "range-parser": "~1.2.1", + "statuses": "~2.0.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/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/serve-static": { + "version": "1.16.3", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.3.tgz", + "integrity": "sha512-x0RTqQel6g5SY7Lg6ZreMmsOzncHFU7nhnRWkKgWuMTu5NN0DR5oruckMqRvacAN9d5w6ARnRBXl9xhDCgfMeA==", + "dependencies": { + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "~0.19.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" + }, + "node_modules/side-channel": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.1.tgz", + "integrity": "sha512-mjn/0bi/oUURjc5Xl7IaWi/OJJJumuoJFQJfDDyO46+hBWsfaVM65TBHq2eoZBhzl9EchxOijpkbRC8SVBQU0w==", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/statuses": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz", + "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/streamsearch": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz", + "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==", + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/string_decoder/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==" + }, + "node_modules/typescript": { + "version": "5.9.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", + "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/uid-safe": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/uid-safe/-/uid-safe-2.1.5.tgz", + "integrity": "sha512-KPHm4VL5dDXKz01UuEd88Df+KzynaohSL9fBh096KWAxSKZQDI2uBrVqtvRM4rwrIrRRKsdLNML/lnaaVSRioA==", + "dependencies": { + "random-bytes": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/undici-types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz", + "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==", + "dev": true + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + }, + "node_modules/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "engines": { + "node": ">=0.4" + } + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..ec28d7d --- /dev/null +++ b/package.json @@ -0,0 +1,26 @@ +{ + "name": "make-video-site", + "version": "0.1.0", + "description": "영상 업로드 및 저장을 위한 사이트", + "type": "module", + "main": "dist/app.js", + "scripts": { + "build": "tsc -p tsconfig.json", + "start": "node dist/app.js", + "dev": "tsc -p tsconfig.json && node dist/app.js" + }, + "dependencies": { + "ejs": "^3.1.10", + "express": "^4.19.2", + "express-session": "^1.18.0", + "multer": "^1.4.5-lts.1" + }, + "devDependencies": { + "@types/ejs": "^3.1.5", + "@types/express": "^4.17.21", + "@types/express-session": "^1.18.0", + "@types/multer": "^1.4.11", + "@types/node": "^22.5.0", + "typescript": "^5.5.4" + } +} diff --git a/public/dashboard.js b/public/dashboard.js new file mode 100644 index 0000000..7e93449 --- /dev/null +++ b/public/dashboard.js @@ -0,0 +1,80 @@ +(function () { + var ctxMenu = document.getElementById('ctxMenu') + var targetName = null + + function showCtx(x, y) { + ctxMenu.style.left = x + 'px' + ctxMenu.style.top = y + 'px' + ctxMenu.hidden = false + } + function hideCtx() { ctxMenu.hidden = true; targetName = null } + + document.querySelectorAll('.adminFolder').forEach(function (card) { + card.addEventListener('contextmenu', function (e) { + e.preventDefault() + targetName = card.getAttribute('data-name') + showCtx(e.clientX, e.clientY) + }) + }) + + document.addEventListener('click', function (e) { + if (!ctxMenu.contains(e.target)) hideCtx() + }) + + ctxMenu.addEventListener('click', function (e) { + var btn = e.target.closest('button') + if (!btn) return + var action = btn.getAttribute('data-action') + if (action === 'rename') { + var newName = window.prompt('새 폴더 이름', targetName) + if (newName && newName !== targetName) { + fetch('/op/folders/rename', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify({ oldName: targetName, newName: newName }) + }).then(function (r) { return r.json() }).then(function (j) { + if (j.ok) location.reload() + else alert(j.message || '이름 변경 실패') + }) + } + } else if (action === 'delete') { + if (window.confirm('"' + targetName + '" 폴더를 정말 삭제할까요? 안의 영상이 모두 사라집니다.')) { + fetch('/op/folders/delete', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify({ name: targetName }) + }).then(function (r) { return r.json() }).then(function (j) { + if (j.ok) location.reload() + else alert(j.message || '삭제 실패') + }) + } + } + hideCtx() + }) + + // 폴더 추가 모달 + var addBtn = document.getElementById('addFolderBtn') + var modal = document.getElementById('addFolderModal') + var input = document.getElementById('addFolderInput') + var cancelBtn = document.getElementById('addFolderCancel') + var confirmBtn = document.getElementById('addFolderConfirm') + + function openModal() { modal.hidden = false; input.value = ''; setTimeout(function () { input.focus() }, 0) } + function closeModal() { modal.hidden = true } + addBtn.addEventListener('click', openModal) + cancelBtn.addEventListener('click', closeModal) + modal.addEventListener('click', function (e) { if (e.target === modal) closeModal() }) + input.addEventListener('keydown', function (e) { if (e.key === 'Enter') confirmBtn.click() }) + confirmBtn.addEventListener('click', function () { + var name = input.value.trim() + if (!name) return + fetch('/op/folders', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify({ name: name }) + }).then(function (r) { return r.json() }).then(function (j) { + if (j.ok) location.reload() + else alert(j.message || '폴더 생성 실패') + }) + }) +})() diff --git a/public/editor.js b/public/editor.js new file mode 100644 index 0000000..b5773be --- /dev/null +++ b/public/editor.js @@ -0,0 +1,197 @@ +(function () { + var ctx = window.__EDITOR__ || { folder: '', video: null } + var folder = ctx.folder + var video = ctx.video + + var dropZone = document.getElementById('dropZone') + var fileInput = document.getElementById('fileInput') + var ytUrl = document.getElementById('ytUrl') + var ytProbeBtn = document.getElementById('ytProbeBtn') + var ytStartBtn = document.getElementById('ytStartBtn') + var probeInfo = document.getElementById('probeInfo') + var dlProgress = document.getElementById('downloadProgress') + var uploadStatus = document.getElementById('uploadStatus') + + var videoPanel = document.getElementById('videoPanel') + var editVideo = document.getElementById('editVideo') + var titleInput = document.getElementById('titleInput') + var startSec = document.getElementById('startSec') + var endSec = document.getElementById('endSec') + var saveBtn = document.getElementById('saveBtn') + + // 드래그&드롭 + ;['dragenter', 'dragover'].forEach(function (evt) { + dropZone.addEventListener(evt, function (e) { + e.preventDefault(); e.stopPropagation() + dropZone.classList.add('dragOver') + }) + }) + ;['dragleave', 'drop'].forEach(function (evt) { + dropZone.addEventListener(evt, function (e) { + e.preventDefault(); e.stopPropagation() + dropZone.classList.remove('dragOver') + }) + }) + dropZone.addEventListener('drop', function (e) { + var files = e.dataTransfer && e.dataTransfer.files + if (files && files.length > 0) uploadFile(files[0]) + }) + fileInput.addEventListener('change', function () { + if (fileInput.files && fileInput.files[0]) uploadFile(fileInput.files[0]) + }) + + function uploadFile(file) { + var form = new FormData() + form.append('file', file) + form.append('title', titleInput.value || file.name) + uploadStatus.textContent = '업로드 중...' + var xhr = new XMLHttpRequest() + xhr.open('POST', '/op/folder/' + encodeURIComponent(folder) + '/video/upload') + xhr.upload.addEventListener('progress', function (e) { + if (e.lengthComputable) { + var pct = Math.round((e.loaded / e.total) * 100) + uploadStatus.textContent = '업로드 ' + pct + '%' + } + }) + xhr.onload = function () { + try { + var res = JSON.parse(xhr.responseText) + if (res.ok) { + location.href = '/op/folder/' + encodeURIComponent(folder) + '/video/editor?id=' + encodeURIComponent(res.videoId) + } else { + uploadStatus.textContent = '업로드 실패: ' + (res.message || '') + } + } catch (err) { + uploadStatus.textContent = '업로드 실패' + } + } + xhr.onerror = function () { uploadStatus.textContent = '업로드 실패 (네트워크)' } + xhr.send(form) + } + + // YouTube probe + ytProbeBtn.addEventListener('click', function () { + var url = ytUrl.value.trim() + if (!url) return + probeInfo.textContent = '확인 중...' + ytStartBtn.disabled = true + fetch('/op/folder/' + encodeURIComponent(folder) + '/video/youtube/probe', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify({ url: url }) + }).then(function (r) { return r.json() }).then(function (j) { + if (!j.ok) { + probeInfo.textContent = j.message || '확인 실패' + return + } + var p = j.probe + var parts = [ + '제목: ' + p.title, + '길이: ' + formatDuration(p.durationSec) + ] + if (p.filesizeApprox) parts.push('대략 ' + formatSize(p.filesizeApprox)) + if (p.etaSec) parts.push('예상 다운로드: ' + formatDuration(p.etaSec)) + probeInfo.textContent = parts.join(' · ') + if (p.warnOver5min) { + if (!window.confirm('가져오는 데 5분 이상 걸릴 수 있습니다. 진행할까요?\n(다른 페이지에서 작업해도 백그라운드로 계속 진행됩니다.)')) { + return + } + } + if (!titleInput.value) titleInput.value = p.title + ytStartBtn.disabled = false + }).catch(function (e) { + probeInfo.textContent = '확인 실패: ' + e.message + }) + }) + + ytStartBtn.addEventListener('click', function () { + var url = ytUrl.value.trim() + if (!url) return + fetch('/op/folder/' + encodeURIComponent(folder) + '/video/youtube/start', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify({ url: url, title: titleInput.value }) + }).then(function (r) { return r.json() }).then(function (j) { + if (!j.ok) { + probeInfo.textContent = j.message || '시작 실패' + return + } + dlProgress.hidden = false + probeInfo.textContent = '백그라운드 다운로드 시작...' + pollJob(j.jobId, j.videoId) + }) + }) + + function pollJob(jobId, videoId) { + fetch('/op/job/' + encodeURIComponent(jobId)).then(function (r) { return r.json() }).then(function (j) { + if (!j.ok) { + probeInfo.textContent = j.message || '작업을 찾을 수 없음' + return + } + var job = j.job + dlProgress.value = job.progress + probeInfo.textContent = job.message + if (job.status === 'done') { + location.href = '/op/folder/' + encodeURIComponent(folder) + '/video/editor?id=' + encodeURIComponent(videoId) + } else if (job.status === 'error') { + probeInfo.textContent = '실패: ' + (job.error || '') + } else { + setTimeout(function () { pollJob(jobId, videoId) }, 1500) + } + }) + } + + // 트림 컨트롤 - "현재 시점" 버튼 + document.querySelectorAll('[data-set-current]').forEach(function (btn) { + btn.addEventListener('click', function () { + if (!editVideo) return + var which = btn.getAttribute('data-set-current') + var t = editVideo.currentTime.toFixed(2) + if (which === 'start') startSec.value = t + else endSec.value = t + }) + }) + + // 저장 + saveBtn.addEventListener('click', function () { + if (!video) { alert('먼저 영상을 추가하세요.'); return } + var payload = { + id: video.id, + title: titleInput.value, + startSec: Number(startSec.value || 0), + endSec: endSec.value === '' ? null : Number(endSec.value) + } + saveBtn.disabled = true + saveBtn.textContent = '저장 중...' + fetch('/op/folder/' + encodeURIComponent(folder) + '/video/save', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify(payload) + }).then(function (r) { return r.json() }).then(function (j) { + saveBtn.disabled = false + saveBtn.textContent = '저장' + if (j.ok) { + alert(j.note || '저장 완료') + location.href = '/op/folder/' + encodeURIComponent(folder) + } else { + alert(j.message || '저장 실패') + } + }) + }) + + function formatDuration(sec) { + if (!sec || sec <= 0) return '0초' + var s = Math.round(sec) + var h = Math.floor(s / 3600); s = s % 3600 + var m = Math.floor(s / 60); s = s % 60 + if (h > 0) return h + '시간 ' + m + '분 ' + s + '초' + if (m > 0) return m + '분 ' + s + '초' + return s + '초' + } + function formatSize(bytes) { + if (bytes < 1024) return bytes + ' B' + if (bytes < 1024 * 1024) return (bytes / 1024).toFixed(1) + ' KB' + if (bytes < 1024 * 1024 * 1024) return (bytes / 1024 / 1024).toFixed(1) + ' MB' + return (bytes / 1024 / 1024 / 1024).toFixed(2) + ' GB' + } +})() diff --git a/public/folder.js b/public/folder.js new file mode 100644 index 0000000..4da78e5 --- /dev/null +++ b/public/folder.js @@ -0,0 +1,59 @@ +(function () { + var folder = (window.__OP__ || {}).folder + var ctxMenu = document.getElementById('ctxMenu') + var targetId = null + var targetTitle = null + + function showCtx(x, y) { + ctxMenu.style.left = x + 'px' + ctxMenu.style.top = y + 'px' + ctxMenu.hidden = false + } + function hideCtx() { ctxMenu.hidden = true } + + document.querySelectorAll('.adminVideo').forEach(function (card) { + card.addEventListener('contextmenu', function (e) { + e.preventDefault() + targetId = card.getAttribute('data-id') + targetTitle = card.getAttribute('data-title') + showCtx(e.clientX, e.clientY) + }) + }) + + document.addEventListener('click', function (e) { + if (!ctxMenu.contains(e.target)) hideCtx() + }) + + ctxMenu.addEventListener('click', function (e) { + var btn = e.target.closest('button') + if (!btn) return + var action = btn.getAttribute('data-action') + if (action === 'edit') { + location.href = '/op/folder/' + encodeURIComponent(folder) + '/video/editor?id=' + encodeURIComponent(targetId) + } else if (action === 'rename') { + var t = window.prompt('새 영상 제목', targetTitle) + if (t && t !== targetTitle) { + fetch('/op/folder/' + encodeURIComponent(folder) + '/video/rename', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify({ id: targetId, title: t }) + }).then(function (r) { return r.json() }).then(function (j) { + if (j.ok) location.reload() + else alert(j.message || '이름 변경 실패') + }) + } + } else if (action === 'delete') { + if (window.confirm('"' + targetTitle + '" 영상을 정말 삭제할까요?')) { + fetch('/op/folder/' + encodeURIComponent(folder) + '/video/delete', { + method: 'POST', + headers: { 'content-type': 'application/json' }, + body: JSON.stringify({ id: targetId }) + }).then(function (r) { return r.json() }).then(function (j) { + if (j.ok) location.reload() + else alert(j.message || '삭제 실패') + }) + } + } + hideCtx() + }) +})() diff --git a/public/player.js b/public/player.js new file mode 100644 index 0000000..a26ef6c --- /dev/null +++ b/public/player.js @@ -0,0 +1,51 @@ +(function () { + var overlay = document.getElementById('playerOverlay') + var closeBtn = document.getElementById('playerClose') + var video = document.getElementById('playerVideo') + var titleEl = document.getElementById('playerTitle') + + function openPlayer(videoId, title) { + // 스펙: /player/:videoId 로 이동한 것처럼 동작하면서 내부 팝업으로 띄운다. + // pushState 로 URL 만 바꿔, 새로고침/직접접근 시 player.ejs 가 응답한다. + history.pushState({ player: true, videoId: videoId }, '', '/player/' + encodeURIComponent(videoId)) + titleEl.textContent = title || '' + video.src = '/api/video/' + encodeURIComponent(videoId) + '/file' + overlay.hidden = false + video.play().catch(function () { /* 자동재생 막힘 무시 */ }) + } + + function closePlayer() { + overlay.hidden = true + video.pause() + video.removeAttribute('src') + video.load() + // 폴더 페이지로 되돌리기 + if (history.state && history.state.player) { + history.back() + } + } + + document.querySelectorAll('.videoCard').forEach(function (card) { + card.addEventListener('click', function () { + var id = card.getAttribute('data-video-id') + var title = card.querySelector('.videoTitle') + openPlayer(id, title ? title.textContent : '') + }) + }) + + if (closeBtn) closeBtn.addEventListener('click', closePlayer) + if (overlay) overlay.addEventListener('click', function (e) { + if (e.target === overlay) closePlayer() + }) + window.addEventListener('popstate', function () { + if (!overlay.hidden) { + overlay.hidden = true + video.pause() + video.removeAttribute('src') + video.load() + } + }) + document.addEventListener('keydown', function (e) { + if (e.key === 'Escape' && !overlay.hidden) closePlayer() + }) +})() diff --git a/public/styles.css b/public/styles.css new file mode 100644 index 0000000..688b10e --- /dev/null +++ b/public/styles.css @@ -0,0 +1,234 @@ +:root { + color-scheme: dark; + --bg: #0d1117; + --bg-alt: #161b22; + --bg-card: #1f242c; + --border: #30363d; + --text: #e6edf3; + --text-muted: #8b949e; + --accent: #2f81f7; + --accent-hover: #1f6feb; + --danger: #f85149; + --success: #3fb950; +} + +* { box-sizing: border-box; } +html, body { margin: 0; padding: 0; } + +body.siteBody { + font-family: 'Pretendard', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; + background: var(--bg); + color: var(--text); + min-height: 100vh; +} +body.siteBody.centerLayout { display: flex; align-items: center; justify-content: center; } + +.pageWrap { max-width: 1200px; margin: 0 auto; padding: 32px 24px 80px; } + +/* nav */ +.topNav, .publicNav { + display: flex; align-items: center; justify-content: space-between; + padding: 16px 32px; background: var(--bg-alt); + border-bottom: 1px solid var(--border); +} +.navBrand { + display: inline-flex; align-items: center; gap: 10px; + text-decoration: none; color: inherit; font-weight: 600; +} +.navLogo { font-size: 22px; } +.navTitle { font-size: 16px; } +.navUser { position: relative; } +.navUserButton { + background: transparent; border: 1px solid var(--border); color: var(--text); + padding: 8px 14px; border-radius: 8px; cursor: pointer; font-size: 14px; +} +.navUserButton:hover { background: var(--bg-card); } +.navUserMenu { + position: absolute; right: 0; top: calc(100% + 6px); + background: var(--bg-card); border: 1px solid var(--border); + border-radius: 10px; padding: 8px; min-width: 160px; + box-shadow: 0 12px 24px rgba(0,0,0,0.4); z-index: 50; +} +.navUserMenu form { margin: 0; } +.dangerLink { + background: transparent; border: none; color: var(--danger); cursor: pointer; + padding: 6px 8px; font-size: 14px; width: 100%; text-align: left; +} +.dangerLink:hover { background: rgba(248,81,73,0.1); border-radius: 6px; } + +/* generic */ +.hero h1 { margin: 8px 0 8px; font-size: 30px; } +.hero p { color: var(--text-muted); margin: 0 0 24px; } +.muted { color: var(--text-muted); font-size: 13px; text-decoration: none; } + +.primaryButton, .secondaryButton, .dangerButton { + border-radius: 8px; padding: 9px 16px; font-size: 14px; cursor: pointer; + border: 1px solid transparent; text-decoration: none; display: inline-block; +} +.primaryButton { background: var(--accent); color: white; } +.primaryButton:hover { background: var(--accent-hover); } +.secondaryButton { background: var(--bg-card); color: var(--text); border-color: var(--border); } +.secondaryButton:hover { background: var(--bg-alt); } +.dangerButton { background: var(--danger); color: white; } + +.dashboardHeader { + display: flex; justify-content: space-between; align-items: flex-end; + margin-bottom: 24px; gap: 16px; flex-wrap: wrap; +} +.dashboardActions { display: flex; gap: 8px; flex-wrap: wrap; } + +/* folder grid */ +.folderGrid { + display: grid; grid-template-columns: repeat(auto-fill, minmax(180px, 1fr)); + gap: 16px; +} +.folderCard { + background: var(--bg-card); border: 1px solid var(--border); + border-radius: 12px; padding: 18px; text-decoration: none; color: var(--text); + display: flex; flex-direction: column; align-items: center; gap: 8px; + cursor: pointer; transition: transform .08s ease; +} +.folderCard:hover { transform: translateY(-2px); border-color: var(--accent); } +.folderCardLink { + display: flex; flex-direction: column; align-items: center; gap: 8px; + text-decoration: none; color: inherit; +} +.folderIcon { font-size: 40px; } +.folderName { font-size: 15px; word-break: break-all; text-align: center; } + +/* video grid */ +.videoGrid { + display: grid; grid-template-columns: repeat(auto-fill, minmax(220px, 1fr)); + gap: 18px; +} +.videoCard { + background: var(--bg-card); border: 1px solid var(--border); + border-radius: 10px; overflow: hidden; cursor: pointer; color: var(--text); + display: flex; flex-direction: column; padding: 0; text-align: left; + font: inherit; +} +.videoCard:hover { border-color: var(--accent); } +.videoThumb { + aspect-ratio: 16/9; background: linear-gradient(135deg, #1f242c, #0d1117); + display: flex; align-items: center; justify-content: center; + font-size: 36px; color: var(--accent); +} +.videoTitle { padding: 10px 12px; font-size: 14px; } + +/* login */ +.loginCard { + background: var(--bg-card); border: 1px solid var(--border); + border-radius: 14px; padding: 32px; min-width: 320px; +} +.loginForm { display: flex; flex-direction: column; gap: 14px; } +.loginForm label { display: flex; flex-direction: column; gap: 6px; } +.loginForm input { + background: var(--bg); border: 1px solid var(--border); border-radius: 8px; + color: var(--text); padding: 10px 12px; font-size: 14px; +} +.errorBanner { + background: rgba(248,81,73,0.15); border: 1px solid var(--danger); + color: #ffb1ab; padding: 10px 12px; border-radius: 8px; font-size: 13px; +} + +/* context menu */ +.ctxMenu { + position: fixed; z-index: 100; background: var(--bg-card); + border: 1px solid var(--border); border-radius: 8px; padding: 6px; + box-shadow: 0 12px 24px rgba(0,0,0,0.4); min-width: 160px; +} +.ctxMenu button { + display: block; width: 100%; text-align: left; background: transparent; + border: none; color: var(--text); padding: 8px 10px; font-size: 13px; + border-radius: 6px; cursor: pointer; +} +.ctxMenu button:hover { background: var(--bg-alt); } + +/* modal */ +.modalOverlay { + position: fixed; inset: 0; background: rgba(0,0,0,0.6); + display: flex; align-items: center; justify-content: center; z-index: 200; +} +.modalCard { + background: var(--bg-card); border: 1px solid var(--border); + border-radius: 14px; padding: 24px; min-width: 320px; +} +.modalCard h2 { margin: 0 0 16px; font-size: 18px; } +.modalCard label { display: flex; flex-direction: column; gap: 6px; } +.modalCard input[type="text"] { + background: var(--bg); border: 1px solid var(--border); border-radius: 8px; + color: var(--text); padding: 10px 12px; font-size: 14px; +} +.modalActions { display: flex; justify-content: flex-end; gap: 8px; margin-top: 16px; } + +/* player modal */ +.playerOverlay { + position: fixed; inset: 0; background: rgba(0,0,0,0.85); + display: flex; align-items: center; justify-content: center; z-index: 300; +} +.playerModal { + position: relative; background: #000; border-radius: 12px; + width: min(960px, 92vw); max-height: 92vh; padding: 16px; + display: flex; flex-direction: column; gap: 10px; +} +.playerClose { + position: absolute; top: 8px; right: 12px; background: transparent; + border: none; color: white; font-size: 22px; cursor: pointer; z-index: 1; +} +.playerTitle { color: white; font-size: 16px; margin-right: 36px; } +.playerModal video { width: 100%; max-height: 80vh; background: #000; } + +.standalonePlayer { background: #000; border-radius: 10px; padding: 12px; } +.standalonePlayer video { width: 100%; max-height: 80vh; } + +/* editor */ +.editorNav { + display: flex; justify-content: space-between; align-items: center; + padding: 12px 24px; background: var(--bg-alt); + border-bottom: 1px solid var(--border); gap: 12px; +} +.editorNavLeft { display: flex; align-items: center; gap: 16px; flex: 1; } +.editorNavRight { display: flex; gap: 8px; } +.titleInput { + background: var(--bg); border: 1px solid var(--border); border-radius: 8px; + color: var(--text); padding: 8px 12px; font-size: 15px; min-width: 280px; + flex: 1; max-width: 480px; +} +.editorMain { padding: 24px; } +.editorStage { max-width: 1200px; margin: 0 auto; } +.dropZone { + background: var(--bg-card); border: 2px dashed var(--border); + border-radius: 12px; padding: 40px; text-align: center; + display: flex; flex-direction: column; gap: 14px; align-items: center; +} +.dropZone.dragOver { border-color: var(--accent); background: rgba(47,129,247,0.08); } +.dropTitle { font-size: 18px; margin: 0; } +.ytRow { + display: flex; gap: 8px; width: 100%; max-width: 640px; flex-wrap: wrap; +} +.ytRow input { + flex: 1; background: var(--bg); border: 1px solid var(--border); + border-radius: 8px; color: var(--text); padding: 10px 12px; font-size: 14px; + min-width: 240px; +} +#downloadProgress { width: 100%; max-width: 640px; height: 8px; } + +.videoPanel { display: flex; flex-direction: column; gap: 16px; } +.videoPanel video { + width: 100%; max-height: 60vh; background: #000; border-radius: 10px; +} +.trimControls { + background: var(--bg-card); border: 1px solid var(--border); + border-radius: 10px; padding: 16px; + display: flex; flex-wrap: wrap; gap: 18px; +} +.trimControls label { + display: flex; flex-direction: column; gap: 6px; font-size: 13px; + color: var(--text-muted); +} +.trimControls input { + background: var(--bg); border: 1px solid var(--border); border-radius: 8px; + color: var(--text); padding: 8px 10px; font-size: 14px; width: 180px; +} +.adminFolder { position: relative; } +.adminVideo { position: relative; } diff --git a/src/app.ts b/src/app.ts new file mode 100644 index 0000000..59b29f6 --- /dev/null +++ b/src/app.ts @@ -0,0 +1,70 @@ +import express from 'express' +import session from 'express-session' +import path from 'node:path' +import fsp from 'node:fs/promises' +import { dataDir, foldersDir, jobsDir, tmpDir, publicDir, viewsDir } from './paths.js' +import { publicRouter } from './routes/public.js' +import { opRouter } from './routes/op.js' + +async function ensureDirs(): Promise { + for (const dir of [dataDir, foldersDir, jobsDir, tmpDir]) { + await fsp.mkdir(dir, { recursive: true }) + } +} + +async function main(): Promise { + await ensureDirs() + + const PORT = Number(process.env.PORT ?? 3000) + const HOST = process.env.HOST ?? '127.0.0.1' + + const app = express() + app.set('view engine', 'ejs') + app.set('views', viewsDir) + app.set('trust proxy', 1) + + app.use(express.urlencoded({ extended: true })) + app.use(express.json({ limit: '4mb' })) + + app.use(session({ + secret: process.env.SESSION_SECRET ?? 'make-video-site-dev-secret', + resave: false, + saveUninitialized: false, + cookie: { + httpOnly: true, + sameSite: 'lax', + maxAge: 1000 * 60 * 60 * 8 + } + })) + + // account.json 직접 노출 차단 + app.use((req, res, next) => { + if (/^\/account\.json/i.test(req.path)) { + res.status(404).send('Not Found') + return + } + next() + }) + + app.use('/static', express.static(publicDir)) + + app.use('/', publicRouter) + app.use('/', opRouter) + + app.use((err: unknown, _req: express.Request, res: express.Response, _next: express.NextFunction) => { + console.error(err) + const message = err instanceof Error ? err.message : '알 수 없는 오류' + if (res.headersSent) return + res.status(500).send(`서버 오류: ${message}`) + }) + + app.listen(PORT, HOST, () => { + console.log(`[server] http://${HOST}:${PORT}`) + console.log(`[server] views: ${path.relative(process.cwd(), viewsDir)}`) + }) +} + +main().catch((err) => { + console.error(err) + process.exit(1) +}) diff --git a/src/auth.ts b/src/auth.ts new file mode 100644 index 0000000..6ab9d8b --- /dev/null +++ b/src/auth.ts @@ -0,0 +1,19 @@ +import type { Request, Response, NextFunction } from 'express' + +declare module 'express-session' { + interface SessionData { + userId?: string + } +} + +export function requireAuth(req: Request, res: Response, next: NextFunction): void { + if (req.session?.userId) { + next() + return + } + if (req.method === 'GET') { + res.redirect('/op') + return + } + res.status(401).json({ ok: false, message: '인증이 필요합니다.' }) +} diff --git a/src/editor.ts b/src/editor.ts new file mode 100644 index 0000000..40de2bd --- /dev/null +++ b/src/editor.ts @@ -0,0 +1,93 @@ +import { spawn, spawnSync } from 'node:child_process' +import { promises as fs } from 'node:fs' +import path from 'node:path' +import { videoDir, loadVideoMeta, saveVideoMeta, type VideoTrim } from './store.js' + +export class FfmpegUnavailableError extends Error { + constructor() { + super('ffmpeg 가 설치되어 있지 않습니다. ffmpeg 를 PATH 에 설치한 뒤 다시 시도해 주세요.') + } +} + +let resolvedFfmpegPath: string | null = null + +export function getFfmpegPath(): string { + if (resolvedFfmpegPath) return resolvedFfmpegPath + const r = spawnSync('ffmpeg', ['-version']) + if (r.status === 0) { + resolvedFfmpegPath = 'ffmpeg' + return 'ffmpeg' + } + throw new FfmpegUnavailableError() +} + +/** + * 원본 파일을 그대로 둔 채 trim 결과를 edited. 로 저장한다. + * stream copy 를 우선 시도해 빠르게 자르고, 실패하면 재인코딩. + */ +export async function applyTrimToVideo( + folder: string, + videoId: string, + trim: VideoTrim +): Promise { + const bin = getFfmpegPath() + const meta = await loadVideoMeta(folder, videoId) + if (!meta) throw new Error('비디오를 찾을 수 없습니다.') + const dir = videoDir(folder, videoId) + const inputPath = path.join(dir, meta.originalFile) + await fs.access(inputPath) + + const ext = path.extname(meta.originalFile) || '.mp4' + const outName = `edited${ext}` + const outPath = path.join(dir, outName) + const tmpPath = outPath + '.tmp' + ext + + const startSec = Math.max(0, Number(trim.startSec) || 0) + const endSec = trim.endSec == null ? null : Math.max(startSec, Number(trim.endSec)) + + const baseArgs = ['-y', '-ss', String(startSec)] + if (endSec !== null) baseArgs.push('-to', String(endSec)) + baseArgs.push('-i', inputPath) + + // 시도 1: stream copy (빠름) + const copyArgs = [...baseArgs, '-c', 'copy', '-movflags', '+faststart', tmpPath] + let ok = await runFfmpeg(bin, copyArgs) + if (!ok) { + // 시도 2: 재인코딩 + const encArgs = [ + ...baseArgs, + '-c:v', 'libx264', '-preset', 'veryfast', '-crf', '23', + '-c:a', 'aac', '-b:a', '128k', + '-movflags', '+faststart', + tmpPath + ] + ok = await runFfmpeg(bin, encArgs) + if (!ok) throw new Error('ffmpeg trim 실패') + } + await fs.rename(tmpPath, outPath) + + meta.editedFile = outName + meta.trim = { startSec, endSec } + await saveVideoMeta(folder, meta) + return outName +} + +function runFfmpeg(bin: string, args: string[]): Promise { + return new Promise((resolve) => { + const child = spawn(bin, args) + let stderr = '' + child.stderr.on('data', (c) => { + stderr = (stderr + c.toString()).slice(-2000) + }) + child.on('error', () => resolve(false)) + child.on('close', (code) => { + if (code === 0) { + resolve(true) + } else { + // 디버그용 stderr 만 콘솔로 + console.error('[ffmpeg] failed:', stderr.split('\n').slice(-5).join('\n')) + resolve(false) + } + }) + }) +} diff --git a/src/paths.ts b/src/paths.ts new file mode 100644 index 0000000..46d0f14 --- /dev/null +++ b/src/paths.ts @@ -0,0 +1,19 @@ +import path from 'node:path' +import { fileURLToPath } from 'node:url' + +// CommonJS/ESM 둘 다 호환되도록 import.meta 가 있을 때만 사용. +// tsc(NodeNext) + .js 임포트로 dist 가 ESM 으로 출력되므로 import.meta.url 이 살아있다. +const here = fileURLToPath(import.meta.url) +const hereDir = path.dirname(here) + +// dist/ 또는 src/ 에서 실행되어도 동일하게 프로젝트 루트를 가리키도록 한다. +export const projectRoot = path.resolve(hereDir, '..') + +export const dataDir = path.join(projectRoot, 'data') +export const foldersDir = path.join(dataDir, 'folders') +export const jobsDir = path.join(dataDir, 'jobs') +export const tmpDir = path.join(dataDir, 'tmp') + +export const viewsDir = path.join(projectRoot, 'views') +export const publicDir = path.join(projectRoot, 'public') +export const accountJsonPath = path.join(projectRoot, 'account.json') diff --git a/src/routes/op.ts b/src/routes/op.ts new file mode 100644 index 0000000..79356ec --- /dev/null +++ b/src/routes/op.ts @@ -0,0 +1,300 @@ +import { Router } from 'express' +import path from 'node:path' +import multer from 'multer' +import { promises as fs } from 'node:fs' +import { requireAuth } from '../auth.js' +import { + createFolder, + deleteFolder, + deleteVideo, + folderPath, + listFolders, + listVideos, + loadVideoMeta, + moveUploadIntoVideo, + newVideoId, + readAccounts, + renameFolder, + sanitizeFolderName, + saveVideoMeta, + type VideoMeta +} from '../store.js' +import { tmpDir } from '../paths.js' +import { + YtDlpUnavailableError, + getJob, + probeYoutube, + startYoutubeDownload +} from '../youtube.js' +import { FfmpegUnavailableError, applyTrimToVideo } from '../editor.js' + +export const opRouter = Router() + +const upload = multer({ + dest: tmpDir, + limits: { fileSize: 4 * 1024 * 1024 * 1024 } // 4GB +}) + +function pickStr(v: unknown): string { + if (Array.isArray(v)) return typeof v[0] === 'string' ? v[0] : '' + return typeof v === 'string' ? v : '' +} + +opRouter.get('/op', (req, res) => { + if (req.session?.userId) { + res.redirect('/op/dashboard') + return + } + res.render('op/login', { error: null }) +}) + +opRouter.post('/op', async (req, res, next) => { + try { + const password = pickStr(req.body.password) + const accounts = await readAccounts() + const matched = accounts.find((a) => a.password === password) + if (!matched) { + res.status(401).render('op/login', { error: '비밀번호가 올바르지 않습니다.' }) + return + } + req.session.userId = matched.id + res.redirect('/op/dashboard') + } catch (err) { + next(err) + } +}) + +opRouter.post('/op/logout', (req, res) => { + req.session.destroy(() => { + res.redirect('/op') + }) +}) + +opRouter.get('/op/dashboard', requireAuth, async (req, res, next) => { + try { + const folders = await listFolders() + res.render('op/dashboard', { userId: req.session.userId, folders }) + } catch (err) { + next(err) + } +}) + +opRouter.post('/op/folders', requireAuth, async (req, res, next) => { + try { + const name = pickStr(req.body.name) + const safe = await createFolder(name) + res.json({ ok: true, name: safe }) + } catch (err) { + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) + +opRouter.post('/op/folders/rename', requireAuth, async (req, res) => { + try { + const oldName = pickStr(req.body.oldName) + const newName = pickStr(req.body.newName) + const result = await renameFolder(oldName, newName) + res.json({ ok: true, name: result }) + } catch (err) { + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) + +opRouter.post('/op/folders/delete', requireAuth, async (req, res) => { + try { + const name = pickStr(req.body.name) + await deleteFolder(name) + res.json({ ok: true }) + } catch (err) { + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) + +opRouter.get('/op/folder/:name', requireAuth, async (req, res, next) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) { + res.status(404).send('폴더를 찾을 수 없습니다.') + return + } + try { + await fs.access(folderPath(safe)) + } catch { + res.status(404).send('폴더를 찾을 수 없습니다.') + return + } + const videos = await listVideos(safe) + res.render('op/folder', { userId: req.session.userId, folder: safe, videos }) + } catch (err) { + next(err) + } +}) + +opRouter.get('/op/folder/:name/video/editor', requireAuth, async (req, res, next) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) { + res.status(404).send('폴더를 찾을 수 없습니다.') + return + } + const videoId = typeof req.query.id === 'string' ? req.query.id : null + let video: VideoMeta | null = null + if (videoId) { + video = await loadVideoMeta(safe, videoId) + } + res.render('op/editor', { + userId: req.session.userId, + folder: safe, + video + }) + } catch (err) { + next(err) + } +}) + +opRouter.post('/op/folder/:name/video/rename', requireAuth, async (req, res) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + const id = pickStr(req.body.id) + const title = pickStr(req.body.title).trim() + if (!title) throw new Error('제목을 입력해 주세요.') + const meta = await loadVideoMeta(safe, id) + if (!meta) throw new Error('영상을 찾을 수 없습니다.') + meta.title = title + await saveVideoMeta(safe, meta) + res.json({ ok: true }) + } catch (err) { + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) + +opRouter.post('/op/folder/:name/video/delete', requireAuth, async (req, res) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + const id = pickStr(req.body.id) + await deleteVideo(safe, id) + res.json({ ok: true }) + } catch (err) { + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) + +// 업로드: 단일 파일. multipart/form-data, fields: title, file +opRouter.post( + '/op/folder/:name/video/upload', + requireAuth, + upload.single('file'), + async (req, res) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + const file = req.file + if (!file) throw new Error('파일이 없습니다.') + const title = pickStr(req.body?.title).trim() || file.originalname + const ext = (path.extname(file.originalname) || '.mp4').toLowerCase() + const videoId = newVideoId() + const destName = `original${ext}` + await moveUploadIntoVideo(safe, videoId, file.path, destName) + const now = new Date().toISOString() + const meta = { + id: videoId, + title, + originalFile: destName, + editedFile: null, + durationSec: null, + sourceType: 'upload' as const, + sourceUrl: null, + trim: null, + createdAt: now, + updatedAt: now + } + await saveVideoMeta(safe, meta) + res.json({ ok: true, videoId, folder: safe }) + } catch (err) { + res.status(400).json({ ok: false, message: (err as Error).message }) + } + } +) + +// 유튜브 프로브: 다운받기 전에 길이/사이즈/예상시간/5분초과경고 +opRouter.post('/op/folder/:name/video/youtube/probe', requireAuth, async (req, res) => { + try { + const url = pickStr(req.body.url).trim() + if (!url) throw new Error('URL 을 입력해 주세요.') + const probe = await probeYoutube(url) + res.json({ ok: true, probe }) + } catch (err) { + if (err instanceof YtDlpUnavailableError) { + res.status(503).json({ ok: false, message: err.message, code: 'NO_YTDLP' }) + return + } + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) + +opRouter.post('/op/folder/:name/video/youtube/start', requireAuth, async (req, res) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + const url = pickStr(req.body.url).trim() + const title = pickStr(req.body.title).trim() || undefined + if (!url) throw new Error('URL 을 입력해 주세요.') + const job = await startYoutubeDownload({ folder: safe, url, title }) + res.json({ ok: true, jobId: job.id, videoId: job.videoId }) + } catch (err) { + if (err instanceof YtDlpUnavailableError) { + res.status(503).json({ ok: false, message: err.message, code: 'NO_YTDLP' }) + return + } + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) + +opRouter.get('/op/job/:id', requireAuth, (req, res) => { + const job = getJob(req.params.id) + if (!job) { + res.status(404).json({ ok: false, message: '작업을 찾을 수 없습니다.' }) + return + } + res.json({ ok: true, job }) +}) + +// 편집 저장: trim 정보를 받아 ffmpeg 로 edited. 생성. 원본은 그대로 보존. +opRouter.post('/op/folder/:name/video/save', requireAuth, async (req, res) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + const id = pickStr(req.body.id) + const title = pickStr(req.body.title).trim() + const startSec = Number(req.body.startSec ?? 0) || 0 + const endRaw = req.body.endSec + const endSec = + endRaw === null || endRaw === undefined || endRaw === '' + ? null + : Number(endRaw) + const meta = await loadVideoMeta(safe, id) + if (!meta) throw new Error('영상을 찾을 수 없습니다.') + if (title) meta.title = title + meta.trim = { startSec, endSec: endSec == null || Number.isNaN(endSec) ? null : endSec } + await saveVideoMeta(safe, meta) + // ffmpeg 가 없으면 trim 정보만 저장하고 안내. + try { + const outName = await applyTrimToVideo(safe, id, meta.trim) + res.json({ ok: true, editedFile: outName, note: '편집본 저장 완료' }) + } catch (err) { + if (err instanceof FfmpegUnavailableError) { + res.json({ + ok: true, + editedFile: null, + note: 'ffmpeg 가 설치되지 않아 편집본을 만들지 못했습니다. trim 설정만 저장됐습니다.' + }) + return + } + throw err + } + } catch (err) { + res.status(400).json({ ok: false, message: (err as Error).message }) + } +}) diff --git a/src/routes/public.ts b/src/routes/public.ts new file mode 100644 index 0000000..a252c70 --- /dev/null +++ b/src/routes/public.ts @@ -0,0 +1,94 @@ +import { Router } from 'express' +import path from 'node:path' +import { promises as fs } from 'node:fs' +import { + findVideoAnywhere, + folderPath, + listFolders, + listVideos, + loadVideoMeta, + sanitizeFolderName, + videoDir, + videoFileFsPath +} from '../store.js' + +export const publicRouter = Router() + +publicRouter.get('/', async (_req, res, next) => { + try { + const folders = await listFolders() + res.render('index', { folders }) + } catch (err) { + next(err) + } +}) + +publicRouter.get('/folder/:name', async (req, res, next) => { + try { + const safe = sanitizeFolderName(req.params.name) + if (!safe) { + res.status(404).send('폴더를 찾을 수 없습니다.') + return + } + // 존재 확인 + try { + await fs.access(folderPath(safe)) + } catch { + res.status(404).send('폴더를 찾을 수 없습니다.') + return + } + const videos = await listVideos(safe) + res.render('folder', { folder: safe, videos, isAdmin: false }) + } catch (err) { + next(err) + } +}) + +publicRouter.get('/player/:videoId', async (req, res, next) => { + try { + const found = await findVideoAnywhere(req.params.videoId) + if (!found) { + res.status(404).send('영상을 찾을 수 없습니다.') + return + } + res.render('player', { folder: found.folder, video: found.meta }) + } catch (err) { + next(err) + } +}) + +/** 영상 파일 스트리밍. ?edited=1 이면 편집본을, 아니면 원본을 보낸다. */ +publicRouter.get('/api/video/:videoId/file', async (req, res, next) => { + try { + const found = await findVideoAnywhere(req.params.videoId) + if (!found) { + res.status(404).end() + return + } + const wantEdited = req.query.edited === '1' || req.query.edited === 'true' + const fileName = + wantEdited && found.meta.editedFile ? found.meta.editedFile : found.meta.originalFile + if (!fileName || fileName.includes('%(ext)s')) { + res.status(404).end() + return + } + const fsPath = videoFileFsPath(found.folder, found.meta.id, fileName) + res.sendFile(fsPath) + } catch (err) { + next(err) + } +}) + +/** 비디오 메타 조회 (플레이어/관리자 양쪽에서 사용) */ +publicRouter.get('/api/video/:videoId', async (req, res, next) => { + try { + const found = await findVideoAnywhere(req.params.videoId) + if (!found) { + res.status(404).json({ ok: false, message: '영상을 찾을 수 없습니다.' }) + return + } + res.json({ ok: true, folder: found.folder, video: found.meta }) + } catch (err) { + next(err) + } +}) diff --git a/src/store.ts b/src/store.ts new file mode 100644 index 0000000..d55b4c1 --- /dev/null +++ b/src/store.ts @@ -0,0 +1,220 @@ +import { promises as fs, createReadStream, createWriteStream } from 'node:fs' +import path from 'node:path' +import { randomUUID } from 'node:crypto' +import { foldersDir, accountJsonPath } from './paths.js' + +export interface Account { + id: string + password: string +} + +export interface VideoTrim { + startSec: number + endSec: number | null // null = until end +} + +export interface VideoMeta { + id: string + title: string + originalFile: string // relative to // + editedFile: string | null + durationSec: number | null + sourceType: 'upload' | 'youtube' + sourceUrl: string | null + trim: VideoTrim | null + createdAt: string + updatedAt: string +} + +export async function readAccounts(): Promise { + try { + const raw = await fs.readFile(accountJsonPath, 'utf8') + const parsed = JSON.parse(raw) + if (!Array.isArray(parsed)) return [] + return parsed.filter( + (entry): entry is Account => + typeof entry?.id === 'string' && typeof entry?.password === 'string' + ) + } catch (err) { + if ((err as NodeJS.ErrnoException).code === 'ENOENT') return [] + throw err + } +} + +// 폴더/영상 이름에 사용 가능한 문자만 허용. 경로 탈출 방지. +const SAFE_NAME = /^[\p{L}\p{N}_\- ]+$/u + +export function sanitizeFolderName(raw: string): string { + const trimmed = (raw || '').trim() + if (trimmed.length === 0 || trimmed.length > 80) return '' + if (!SAFE_NAME.test(trimmed)) return '' + if (trimmed === '.' || trimmed === '..') return '' + return trimmed +} + +export function isSafeVideoId(id: string): boolean { + return /^[a-zA-Z0-9_-]{8,64}$/.test(id) +} + +export async function ensureFoldersDir(): Promise { + await fs.mkdir(foldersDir, { recursive: true }) +} + +export async function listFolders(): Promise { + await ensureFoldersDir() + const entries = await fs.readdir(foldersDir, { withFileTypes: true }) + return entries + .filter((e) => e.isDirectory()) + .map((e) => e.name) + .filter((name) => sanitizeFolderName(name).length > 0) + .sort((a, b) => a.localeCompare(b, 'ko')) +} + +export async function createFolder(name: string): Promise { + const safe = sanitizeFolderName(name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + const target = path.join(foldersDir, safe) + await fs.mkdir(target, { recursive: false }).catch((err) => { + if (err.code === 'EEXIST') throw new Error('이미 존재하는 폴더입니다.') + throw err + }) + return safe +} + +export async function renameFolder(oldName: string, newName: string): Promise { + const oldSafe = sanitizeFolderName(oldName) + const newSafe = sanitizeFolderName(newName) + if (!oldSafe || !newSafe) throw new Error('폴더 이름이 올바르지 않습니다.') + if (oldSafe === newSafe) return oldSafe + const from = path.join(foldersDir, oldSafe) + const to = path.join(foldersDir, newSafe) + try { + await fs.access(to) + throw new Error('이미 존재하는 폴더입니다.') + } catch (err) { + if ((err as NodeJS.ErrnoException).code !== 'ENOENT') throw err + } + await fs.rename(from, to) + return newSafe +} + +export async function deleteFolder(name: string): Promise { + const safe = sanitizeFolderName(name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + const target = path.join(foldersDir, safe) + await fs.rm(target, { recursive: true, force: true }) +} + +export function folderPath(name: string): string { + const safe = sanitizeFolderName(name) + if (!safe) throw new Error('폴더 이름이 올바르지 않습니다.') + return path.join(foldersDir, safe) +} + +export function videoDir(folder: string, videoId: string): string { + if (!isSafeVideoId(videoId)) throw new Error('비디오 ID가 올바르지 않습니다.') + return path.join(folderPath(folder), videoId) +} + +export function videoMetaPath(folder: string, videoId: string): string { + return path.join(videoDir(folder, videoId), 'meta.json') +} + +export async function listVideos(folder: string): Promise { + const dir = folderPath(folder) + try { + const entries = await fs.readdir(dir, { withFileTypes: true }) + const metas: VideoMeta[] = [] + for (const entry of entries) { + if (!entry.isDirectory()) continue + if (!isSafeVideoId(entry.name)) continue + const meta = await loadVideoMeta(folder, entry.name).catch(() => null) + if (meta) metas.push(meta) + } + metas.sort((a, b) => (b.createdAt < a.createdAt ? -1 : 1)) + return metas + } catch (err) { + if ((err as NodeJS.ErrnoException).code === 'ENOENT') return [] + throw err + } +} + +export async function loadVideoMeta(folder: string, videoId: string): Promise { + try { + const raw = await fs.readFile(videoMetaPath(folder, videoId), 'utf8') + return JSON.parse(raw) as VideoMeta + } catch (err) { + if ((err as NodeJS.ErrnoException).code === 'ENOENT') return null + throw err + } +} + +export async function findVideoAnywhere( + videoId: string +): Promise<{ folder: string; meta: VideoMeta } | null> { + if (!isSafeVideoId(videoId)) return null + const folders = await listFolders() + for (const folder of folders) { + const meta = await loadVideoMeta(folder, videoId).catch(() => null) + if (meta) return { folder, meta } + } + return null +} + +export async function saveVideoMeta(folder: string, meta: VideoMeta): Promise { + const dir = videoDir(folder, meta.id) + await fs.mkdir(dir, { recursive: true }) + meta.updatedAt = new Date().toISOString() + await fs.writeFile(videoMetaPath(folder, meta.id), JSON.stringify(meta, null, 2)) +} + +export async function deleteVideo(folder: string, videoId: string): Promise { + const dir = videoDir(folder, videoId) + await fs.rm(dir, { recursive: true, force: true }) +} + +export function newVideoId(): string { + // URL-safe 22-char id (uuid 압축). + return randomUUID().replace(/-/g, '').slice(0, 24) +} + +export function videoFileFsPath(folder: string, videoId: string, rel: string): string { + // rel 은 meta 에 저장된 파일명. 경로 탈출 방지. + if (rel.includes('/') || rel.includes('\\') || rel.includes('..')) { + throw new Error('잘못된 파일 경로입니다.') + } + return path.join(videoDir(folder, videoId), rel) +} + +/** 업로드 임시 파일을 비디오 디렉토리로 이동. */ +export async function moveUploadIntoVideo( + folder: string, + videoId: string, + tmpFile: string, + destName: string +): Promise { + if (destName.includes('/') || destName.includes('\\') || destName.includes('..')) { + throw new Error('잘못된 파일명입니다.') + } + const dir = videoDir(folder, videoId) + await fs.mkdir(dir, { recursive: true }) + const target = path.join(dir, destName) + try { + await fs.rename(tmpFile, target) + } catch (err) { + if ((err as NodeJS.ErrnoException).code === 'EXDEV') { + // 다른 디바이스에 있으면 copy + unlink + await new Promise((resolve, reject) => { + const r = createReadStream(tmpFile) + const w = createWriteStream(target) + r.on('error', reject) + w.on('error', reject) + w.on('close', () => resolve()) + r.pipe(w) + }) + await fs.unlink(tmpFile).catch(() => undefined) + } else { + throw err + } + } +} diff --git a/src/youtube.ts b/src/youtube.ts new file mode 100644 index 0000000..6122044 --- /dev/null +++ b/src/youtube.ts @@ -0,0 +1,243 @@ +import { spawn, spawnSync } from 'node:child_process' +import { promises as fs } from 'node:fs' +import path from 'node:path' +import { jobsDir, projectRoot } from './paths.js' +import { + loadVideoMeta, + newVideoId, + saveVideoMeta, + sanitizeFolderName, + videoDir, + type VideoMeta +} from './store.js' + +export class YtDlpUnavailableError extends Error { + constructor(message?: string) { + super(message || 'yt-dlp 가 설치되어 있지 않습니다. yt-dlp 를 PATH 에 설치한 뒤 다시 시도해 주세요.') + } +} + +let resolvedYtDlpPath: string | null = null + +export function getYtDlpPath(): string { + if (resolvedYtDlpPath) return resolvedYtDlpPath + // 1) 프로젝트 bin/yt-dlp(.exe) + const localCandidates = + process.platform === 'win32' + ? ['bin/yt-dlp.exe', 'bin/yt-dlp'] + : ['bin/yt-dlp'] + for (const rel of localCandidates) { + const abs = path.join(projectRoot, rel) + const r = spawnSync(abs, ['--version']) + if (r.status === 0) { + resolvedYtDlpPath = abs + return abs + } + } + // 2) PATH 에서 검색 + const r = spawnSync('yt-dlp', ['--version']) + if (r.status === 0) { + resolvedYtDlpPath = 'yt-dlp' + return 'yt-dlp' + } + throw new YtDlpUnavailableError() +} + +export interface ProbeResult { + title: string + durationSec: number + filesizeApprox: number | null + /** 추정 다운로드 ETA (초). filesize_approx / 가정대역폭. 모를 때 null. */ + etaSec: number | null + warnOver5min: boolean +} + +const ASSUMED_BPS = 5 * 1024 * 1024 // 5 MB/s 가정. 실측은 어렵지만 경고 트리거용으로만 사용. + +export async function probeYoutube(url: string): Promise { + const bin = getYtDlpPath() + return new Promise((resolve, reject) => { + const child = spawn(bin, [ + '--no-warnings', + '--skip-download', + '--print', + '%(title)s\n%(duration)s\n%(filesize_approx)s' + ].concat([url])) + let stdout = '' + let stderr = '' + child.stdout.on('data', (chunk) => (stdout += chunk.toString())) + child.stderr.on('data', (chunk) => (stderr += chunk.toString())) + child.on('error', (err) => reject(err)) + child.on('close', (code) => { + if (code !== 0) { + reject(new Error(stderr.trim() || `yt-dlp probe 실패 (code=${code})`)) + return + } + const lines = stdout.trim().split('\n') + const title = lines[0] || '' + const durationSec = Number(lines[1]) || 0 + const sizeRaw = lines[2] + const filesizeApprox = + sizeRaw && sizeRaw !== 'NA' && Number.isFinite(Number(sizeRaw)) + ? Number(sizeRaw) + : null + const etaSec = filesizeApprox ? Math.round(filesizeApprox / ASSUMED_BPS) : null + const warnOver5min = (etaSec ?? 0) > 5 * 60 || durationSec > 60 * 60 + resolve({ title, durationSec, filesizeApprox, etaSec, warnOver5min }) + }) + }) +} + +// ─── 백그라운드 다운로드 잡 ──────────────────────────────────────────────── +export type JobStatus = 'queued' | 'downloading' | 'done' | 'error' + +export interface DownloadJob { + id: string + folder: string + videoId: string + url: string + status: JobStatus + progress: number // 0..100 + message: string + startedAt: string + finishedAt: string | null + outputFile: string | null + error: string | null +} + +const jobs = new Map() + +async function persistJob(job: DownloadJob): Promise { + await fs.mkdir(jobsDir, { recursive: true }) + await fs.writeFile(path.join(jobsDir, `${job.id}.json`), JSON.stringify(job, null, 2)) +} + +export function getJob(id: string): DownloadJob | null { + return jobs.get(id) ?? null +} + +export function listActiveJobs(): DownloadJob[] { + return Array.from(jobs.values()).filter((j) => j.status === 'queued' || j.status === 'downloading') +} + +export interface StartDownloadOpts { + folder: string + url: string + title?: string +} + +/** 백그라운드 yt-dlp 다운로드를 시작. videoId 도 함께 생성/저장한다. */ +export async function startYoutubeDownload(opts: StartDownloadOpts): Promise { + const safeFolder = sanitizeFolderName(opts.folder) + if (!safeFolder) throw new Error('폴더 이름이 올바르지 않습니다.') + const bin = getYtDlpPath() // 없으면 즉시 던짐 + const videoId = newVideoId() + const dir = videoDir(safeFolder, videoId) + await fs.mkdir(dir, { recursive: true }) + + const now = new Date().toISOString() + const meta: VideoMeta = { + id: videoId, + title: opts.title?.trim() || '제목 없음', + originalFile: 'original.%(ext)s', // 다운로드 완료 후 실제 확장자로 갱신 + editedFile: null, + durationSec: null, + sourceType: 'youtube', + sourceUrl: opts.url, + trim: null, + createdAt: now, + updatedAt: now + } + await saveVideoMeta(safeFolder, meta) + + const job: DownloadJob = { + id: newVideoId(), + folder: safeFolder, + videoId, + url: opts.url, + status: 'queued', + progress: 0, + message: '대기 중', + startedAt: now, + finishedAt: null, + outputFile: null, + error: null + } + jobs.set(job.id, job) + void persistJob(job) + setImmediate(() => runJob(job, bin).catch((err) => { + job.status = 'error' + job.error = err instanceof Error ? err.message : String(err) + job.finishedAt = new Date().toISOString() + void persistJob(job) + })) + return job +} + +async function runJob(job: DownloadJob, bin: string): Promise { + job.status = 'downloading' + job.message = '다운로드 시작' + await persistJob(job) + + const dir = videoDir(job.folder, job.videoId) + // yt-dlp 가 다운로드 후 결정한 실제 파일명을 알려주도록 --print after_move:filepath + // 진행률은 --newline + --progress-template 으로 stdout 줄 단위 파싱. + const args = [ + '--no-warnings', + '--no-playlist', + '--newline', + '--progress-template', 'download:PROGRESS %(progress._percent_str)s', + '--print', 'after_move:OUT %(filepath)s', + '-o', path.join(dir, 'original.%(ext)s'), + job.url + ] + return new Promise((resolve, reject) => { + const child = spawn(bin, args) + let outputFile: string | null = null + let stderrTail = '' + child.stdout.on('data', (chunk) => { + const text = chunk.toString() + for (const line of text.split(/\r?\n/)) { + const m = /PROGRESS\s+([\d.]+)%/.exec(line) + if (m) { + job.progress = Math.min(99, Math.round(Number(m[1]))) + job.message = `다운로드 ${job.progress}%` + } + const o = /^OUT\s+(.+)$/.exec(line.trim()) + if (o) outputFile = o[1].trim() + } + }) + child.stderr.on('data', (chunk) => { + stderrTail = (stderrTail + chunk.toString()).slice(-2000) + }) + child.on('error', (err) => reject(err)) + child.on('close', async (code) => { + if (code !== 0) { + reject(new Error(stderrTail.trim() || `yt-dlp 실패 (code=${code})`)) + return + } + // 실제 파일명 결정 + let finalName = 'original' + if (outputFile) { + finalName = path.basename(outputFile) + } else { + // fallback: 디렉토리 내 original.* 찾기 + const entries = await fs.readdir(dir).catch(() => []) + const found = entries.find((n) => n.startsWith('original.')) + if (found) finalName = found + } + const meta = await loadVideoMeta(job.folder, job.videoId) + if (meta) { + meta.originalFile = finalName + await saveVideoMeta(job.folder, meta) + } + job.progress = 100 + job.status = 'done' + job.message = '완료' + job.outputFile = finalName + job.finishedAt = new Date().toISOString() + await persistJob(job) + resolve() + }) + }) +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..b51d36f --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,15 @@ +{ + "compilerOptions": { + "target": "ES2022", + "module": "NodeNext", + "moduleResolution": "NodeNext", + "outDir": "dist", + "rootDir": "src", + "strict": true, + "esModuleInterop": true, + "skipLibCheck": true, + "resolveJsonModule": true, + "forceConsistentCasingInFileNames": true + }, + "include": ["src/**/*.ts"] +} diff --git a/views/folder.ejs b/views/folder.ejs new file mode 100644 index 0000000..fb46d1c --- /dev/null +++ b/views/folder.ejs @@ -0,0 +1,50 @@ + + + + + + <%= folder %> · 비디오 사이트 + + + +
+ + + 비디오 사이트 + + 관리자 +
+ +
+
+ ← 폴더 목록 +

📁 <%= folder %>

+
+ +
+ <% if (videos.length === 0) { %> +

이 폴더에 영상이 없습니다.

+ <% } %> + <% videos.forEach(function (v) { %> + + <% }) %> +
+
+ + + + + + + diff --git a/views/index.ejs b/views/index.ejs new file mode 100644 index 0000000..5d334b4 --- /dev/null +++ b/views/index.ejs @@ -0,0 +1,37 @@ + + + + + + 비디오 사이트 + + + +
+ + + 비디오 사이트 + + 관리자 +
+ +
+
+

폴더

+

저장된 영상 폴더를 선택해 안에 있는 영상을 확인하세요.

+
+ +
+ <% if (folders.length === 0) { %> +

아직 폴더가 없습니다. 관리자 페이지에서 폴더를 추가해 주세요.

+ <% } %> + <% folders.forEach(function (name) { %> + + 📁 + <%= name %> + + <% }) %> +
+
+ + diff --git a/views/op/dashboard.ejs b/views/op/dashboard.ejs new file mode 100644 index 0000000..6ada2fb --- /dev/null +++ b/views/op/dashboard.ejs @@ -0,0 +1,56 @@ + + + + + + 관리자 · 폴더 + + + + <%- include('../partials/navbar', { userId }) %> + +
+
+

폴더

+
+ +
+
+ +
+ <% if (folders.length === 0) { %> +

아직 폴더가 없습니다. 우측 상단에서 폴더를 추가해 주세요.

+ <% } %> + <% folders.forEach(function (name) { %> + + <% }) %> +
+
+ + + + + + + + diff --git a/views/op/editor.ejs b/views/op/editor.ejs new file mode 100644 index 0000000..2ab4dd2 --- /dev/null +++ b/views/op/editor.ejs @@ -0,0 +1,64 @@ + + + + + + 영상 편집 · <%= folder %> + + + +
+ +
+ +
+
+ +
+
+
hidden<% } %>> +

영상 추가

+

파일을 드래그&드롭하거나, 아래에서 직접 선택하거나, YouTube 주소를 붙여넣으세요.

+ +
+ + + +
+

+ +

+
+ +
hidden<% } %>> + +
+ + +

저장하면 ffmpeg 가 원본을 보존한 채 편집본을 만듭니다.

+
+
+
+
+ + + + + diff --git a/views/op/folder.ejs b/views/op/folder.ejs new file mode 100644 index 0000000..52a71f6 --- /dev/null +++ b/views/op/folder.ejs @@ -0,0 +1,50 @@ + + + + + + 관리자 · <%= folder %> + + + + <%- include('../partials/navbar', { userId }) %> + +
+
+
+ ← 폴더 목록 +

📁 <%= folder %>

+
+ +
+ +
+ <% if (videos.length === 0) { %> +

이 폴더에 영상이 없습니다. 우측 상단에서 영상을 추가하세요.

+ <% } %> + <% videos.forEach(function (v) { %> +
+
+
<%= v.title %>
+ <% if (v.sourceType === 'youtube' && !v.originalFile.includes('original.') === false) { %> +
YouTube
+ <% } %> +
+ <% }) %> +
+
+ + + + + + + diff --git a/views/op/login.ejs b/views/op/login.ejs new file mode 100644 index 0000000..d7d6b5c --- /dev/null +++ b/views/op/login.ejs @@ -0,0 +1,24 @@ + + + + + + 관리자 로그인 · 비디오 사이트 + + + +
+

관리자 로그인

+ <% if (error) { %> +

<%= error %>

+ <% } %> +
+ + +
+
+ + diff --git a/views/partials/navbar.ejs b/views/partials/navbar.ejs new file mode 100644 index 0000000..aeacffd --- /dev/null +++ b/views/partials/navbar.ejs @@ -0,0 +1,30 @@ +
+ + + 비디오 사이트 관리 + + +
+ diff --git a/views/player.ejs b/views/player.ejs new file mode 100644 index 0000000..cc17820 --- /dev/null +++ b/views/player.ejs @@ -0,0 +1,29 @@ + + + + + + <%= video.title %> · 재생 + + + +
+ + + 비디오 사이트 + + 폴더로 +
+ +
+
+ ← <%= folder %> +

<%= video.title %>

+
+ +
+ +
+
+ +