From 8ffb9ee8c52eda6d40c205d9e124f8a42730758a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <11900611@qq.com> Date: Thu, 22 Jun 2023 14:48:00 +0800 Subject: [PATCH 01/88] Modified to a One-Click Deploy Version And hide the text title of the icon in the upper right corner of the page (Navbar) --- README.md | 7 +++++++ config/api.config.js | 4 ++-- config/site.config.js | 10 +++++----- src/components/Navbar.tsx | 4 ++-- 4 files changed, 16 insertions(+), 9 deletions(-) diff --git a/README.md b/README.md index d55c8b5d53..bb9893009c 100644 --- a/README.md +++ b/README.md @@ -11,6 +11,13 @@ GitHub Discussions +## Rapid Deploy + +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Fspencerwooo%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) + +> **Unless you already know what to do after clicking this button, it is highly recommended that you read the instructions below first. Thank you.** +> + ## TL;DR Showcase, share, preview, and download files inside *your* OneDrive with onedrive-vercel-index - diff --git a/config/api.config.js b/config/api.config.js index 8d1a25142d..8471aeb720 100644 --- a/config/api.config.js +++ b/config/api.config.js @@ -10,8 +10,8 @@ module.exports = { // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. - clientId: 'd87bcc39-1750-4ca0-ad54-f8d0efbb2735', - obfuscatedClientSecret: 'U2FsdGVkX1830zo3/pFDqaBCVBb37iLw3WnBDWGF9GIB2f4apzv0roemp8Y+iIxI3Ih5ecyukqELQEGzZlYiWg==', + clientId: process.env.NEXT_PUBLIC_CLIENT_ID, + obfuscatedClientSecret: process.env.NEXT_PUBLIC_CLIENT_SECRET, // The redirectUri is the URL that the user will be redirected to after they have authenticated with Microsoft Graph API. // Likewise, you would not need to change redirectUri if you are using your personal Microsoft account with OneDrive International. diff --git a/config/site.config.js b/config/site.config.js index 82d46494d4..9a05c1ce2d 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -7,7 +7,7 @@ module.exports = { // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about // your email being exposed in public. - userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPLE_NAME || 'spencer@spwoo.onmicrosoft.com', + userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPLE_NAME, // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. @@ -17,14 +17,14 @@ module.exports = { kvPrefix: process.env.KV_PREFIX || '', // The name of your website. Present alongside your icon. - title: "Spencer's OneDrive", + title: process.env.NEXT_PUBLIC_SITE_TITLE || 'OneDrive-Vercel-Index', // The folder that you are to share publicly with onedrive-vercel-index. Use '/' if you want to share your root folder. - baseDirectory: '/Public', + baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. - maxItems: 100, + maxItems: 200, // [OPTIONAL] We use Google Fonts natively for font customisations. // You can check and generate the required links and names at https://fonts.google.com. @@ -42,7 +42,7 @@ module.exports = { // [OPTIONAL] This is where you specify the folders that are password protected. It is an array of paths pointing to all // the directories in which you have .password set. Check the documentation for details. - protectedRoutes: ['/🌞 Private folder/u-need-a-password', '/🥟 Some test files/Protected route'], + protectedRoutes: [], // [OPTIONAL] Use "" here if you want to remove this email address from the nav bar. email: 'mailto:spencer.wushangbo@gmail.com', diff --git a/src/components/Navbar.tsx b/src/components/Navbar.tsx index 9a7ac24aae..2877f9e6a7 100644 --- a/src/components/Navbar.tsx +++ b/src/components/Navbar.tsx @@ -99,13 +99,13 @@ const Navbar = () => { className="flex items-center space-x-2 hover:opacity-80 dark:text-white" > - + {/* { // Append link name comments here to add translations // t('Weibo') t(l.name) } - + */} ))} From fc394056546c031e258c92a53622591475577f06 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <11900611@qq.com> Date: Thu, 22 Jun 2023 15:21:19 +0800 Subject: [PATCH 02/88] Update README.md --- README.md | 11 ++++------- 1 file changed, 4 insertions(+), 7 deletions(-) diff --git a/README.md b/README.md index bb9893009c..812f052a13 100644 --- a/README.md +++ b/README.md @@ -11,13 +11,6 @@ GitHub Discussions -## Rapid Deploy - -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Fspencerwooo%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) - -> **Unless you already know what to do after clicking this button, it is highly recommended that you read the instructions below first. Thank you.** -> - ## TL;DR Showcase, share, preview, and download files inside *your* OneDrive with onedrive-vercel-index - @@ -31,6 +24,10 @@ Showcase, share, preview, and download files inside *your* OneDrive with onedriv ## Quick start +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Fspencerwooo%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) + +> **Unless you already know what to do after clicking this button, it is highly recommended that you read the instructions below first. Thank you.** + 🚀 Quick start: [Getting started](https://ovi.swo.moe/docs/getting-started). ## Discussion From 5705876a9f6a4a88d52b13fd9ffee981e96aab22 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <11900611@qq.com> Date: Thu, 22 Jun 2023 15:33:00 +0800 Subject: [PATCH 03/88] Update api.config.js --- config/api.config.js | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/config/api.config.js b/config/api.config.js index 8471aeb720..ccc50bcac6 100644 --- a/config/api.config.js +++ b/config/api.config.js @@ -7,11 +7,20 @@ * - If you are using a E5 Subscription OneDrive for Business account, the direct links of your files are not the same here. * In which case you would need to change directLinkRegex. */ +const clientIdEnv = process.env.NEXT_PUBLIC_CLIENT_ID; +if (!clientIdEnv) { + throw new Error('CLIENT_ID is not defined in api.config.js'); +} +const clientSecretEnv = process.env.NEXT_PUBLIC_CLIENT_SECRET; +if (!clientSecretEnv) { + throw new Error('CLIENT_SECRET is not defined in api.config.js'); +} + module.exports = { // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. - clientId: process.env.NEXT_PUBLIC_CLIENT_ID, - obfuscatedClientSecret: process.env.NEXT_PUBLIC_CLIENT_SECRET, + clientId: clientIdEnv, + obfuscatedClientSecret: clientSecretEnv, // The redirectUri is the URL that the user will be redirected to after they have authenticated with Microsoft Graph API. // Likewise, you would not need to change redirectUri if you are using your personal Microsoft account with OneDrive International. From 345a2d8d97d28edb85ef075ecd2da8495c2b0aa9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <11900611@qq.com> Date: Thu, 22 Jun 2023 15:42:21 +0800 Subject: [PATCH 04/88] Update Navbar.tsx --- src/components/Navbar.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/Navbar.tsx b/src/components/Navbar.tsx index 2877f9e6a7..d4bc15a4bb 100644 --- a/src/components/Navbar.tsx +++ b/src/components/Navbar.tsx @@ -112,7 +112,7 @@ const Navbar = () => { {siteConfig.email && ( - {t('Email')} + {/*{t('Email')}*/} )} From 708a78269ba6641a63bb8a96c2befbebc74d1cfa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <11900611@qq.com> Date: Thu, 22 Jun 2023 15:44:37 +0800 Subject: [PATCH 05/88] Create vercel.json --- vercel.json | 4 ++++ 1 file changed, 4 insertions(+) create mode 100644 vercel.json diff --git a/vercel.json b/vercel.json new file mode 100644 index 0000000000..d90a3916af --- /dev/null +++ b/vercel.json @@ -0,0 +1,4 @@ +{ + "buildCommand": "pnpm build", + "installCommand": "pnpm install" +} \ No newline at end of file From 01ca1512c6c796660530559201c9cf10dbd6be23 Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Sun, 25 Jun 2023 12:19:30 +0800 Subject: [PATCH 06/88] update README --- LICENSE | 3 +- README.md | 163 +++++++++++------------------------------- README.zh-CN.md | 71 ++++++++++++++++++ config/site.config.js | 6 +- 4 files changed, 117 insertions(+), 126 deletions(-) create mode 100644 README.zh-CN.md diff --git a/LICENSE b/LICENSE index 6946f10a00..f1f2ea0e13 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,7 @@ MIT License -Copyright (c) 2021 Spencer Woo +Copyright (c) 2021-2023 Spencer Woo +Copyright (c) 2023 iRedScarf Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/README.md b/README.md index 812f052a13..f9e5efaa64 100644 --- a/README.md +++ b/README.md @@ -1,152 +1,71 @@ -
- onedrive-vercel-index -

onedrive-vercel-index

-

Get started · What's new? · Sponsoring

-

OneDrive public directory listing, powered by Vercel and Next.js

+# OneDrive-Vercel-Index (One-Click Deploy Version) - OneDrive - Next.js - Vercel - Documentation - GitHub Discussions -
+English | [中文简体](./README.zh-CN.md) -## TL;DR +This project is a fork from [spencerwooo/onedrive-vercel-index](https://github.com/spencerwooo/onedrive-vercel-index), based on the archived version from the original author dated June 24, 2023. It includes some minor modifications that allow you to deploy it on Vercel for free, showcasing, sharing, previewing, and downloading your OneDrive files on a webpage. For specific deployment methods, please refer to the instructions below. -Showcase, share, preview, and download files inside *your* OneDrive with onedrive-vercel-index - +> This version has only been tested with an E5 Developer account. Other types of OneDrive accounts need further testing. -- Completely free to host 💸 -- Super fast ⚡ and responsive 💦 -- Takes less than 15 minutes to setup ⏱️ -- Highly customisable ⚒️ +## Demo -🍌 More importantly, we are pretty (●'◡'●) +The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](https://onedrive-index-iredscarf.vercel.app) of this One-Click Deployment version. -## Quick start +![demo](./public/demo.png) -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Fspencerwooo%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +## Getting Started -> **Unless you already know what to do after clicking this button, it is highly recommended that you read the instructions below first. Thank you.** +### Preparations -🚀 Quick start: [Getting started](https://ovi.swo.moe/docs/getting-started). +1. **Setting up the API permissions for your OneDrive account:** -## Discussion + This project retrieves the file list and download links by calling OneDrive's API, so setting up the API permissions for your OneDrive account is essential. Please refer to the [Docs](https://ovi.swo.moe/docs/advanced#modify-api-permissions) written by the original author for retrieval methods. -Please go to our [discussion forum](https://github.com/spencerwooo/onedrive-vercel-index/discussions) for general questions and FAQs, **issues are for bug reports and bug reports only.** Feature requests may or may not be ignored, as [I (@spencerwooo)](https://spencerwoo.com) am the only one maintaining the project, so **I only prioritise features that I use.** + The three API permissions that need to be set up are: `user.read`, `files.read.all`, `offline_access`. -*If you happen to like this project, please give it a star!* :3 +2. **Prepare the five environmental parameters to be filled in during deployment on Vercel:** -*If you really, really like this project, please send money! -> [Sponsors 🤑 and donations 💰](https://ovi.swo.moe/sponsor/ways)* +| Name | Description | Default | Note | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_SITE_TITLE` | Title of the displayed page | `null` | e.g., OneDrive of the Richest Man in Nicaragua | +| `NEXT_PUBLIC_USER_PRINCIPLE_NAME` | Your OneDrive account | `null` | **Case-sensitive** | +| `NEXT_PUBLIC_BASE_DIRECTORY` | The OneDrive directory you want to share | `null` | Format is `/directory-name`, for root directory, fill in `/` | +| `NEXT_PUBLIC_CLIENT_ID` | Client ID of the app you registered in Microsoft Azure | `null` | The one provided by the original author has expired, it's recommended to register your own, which can be valid for up to two years. Retrieval methods are the same as in the [Docs](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) written by the original author | +| `NEXT_PUBLIC_CLIENT_SECRET` | Client Secret of the app you registered in Microsoft Azure | `null` | Retrieval methods are the same as above, note that you need to **AES encrypt the original secret** (can be done as described in the [Docs](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | -## Demo +### Deploying to Vercel -Live demo at [Spencer's OneDrive](https://drive.swo.moe). +**Once you're prepared, you can click the button below to deploy:** -![demo](./public/demo.png) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) -## Features - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
- 👀 File preview - - 💠 List / Grid layouts - - 🎥 Video and audio -
PDF, EPUB, markdown, code, plain textFor previewing images and documents with thumbnailsmp4, mp3, ..., play online or with IINA, PotPlayer ... with subtitles!
- 📄 Office preview - 📝 README.md preview📑 Pagination
docx, pptx, xlsx, ...Also renders code blocks, images with relative links, ...For folders with 200 or more items
🔒 Protected folders⏬ Multi-file download🔎 Native Search
Password protected routes and files. Details here - Compress and download multiple files or folders. - Details here - - Searching through your shared OneDrive files (with some caveats 🥺). - Details here -
- -... and more: - -- Streamlined deployment, without having to get your tokens manually anymore! -- Direct raw-file serving and hosting ... -- Full dark mode support, style and website customisations ... - -> **Note**: This project is focused on showcasing and providing a way for others to download files from your OneDrive. Emphasis on **free** and **serverless**. If you have your own server / need WebDAV / use cloud providers other than OneDrive, checkout [alist](https://github.com/alist-org/alist). +- After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. -## Documentation +- `REDIS_URL`:If you are encountering Redis database for the first time like me, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash), it will automatically fill in the environment variables after project deployment. + +- After `REDIS_URL` is successfully set, redeploy the project again. -Documentation is hosted at [ovi.swo.moe](https://ovi.swo.moe/). +**After successful deployment, when you visit your `onedrive-vercel-index` page for the first time, it will guide you to perform OAuth authentication (quite simple). For details, please refer to the [Instructions](https://ovi.swo.moe/zh/docs/getting-started#authentication) written by the original author.** -- [How can I get started and deploy?](https://ovi.swo.moe/docs/getting-started) -- [How can I configure ... ?](https://ovi.swo.moe/docs/custom-configs) -- Where is feature ... ? - - [Docs - Password protected folders](https://ovi.swo.moe/docs/features/protected-folders) - - [Docs - Multi-file and folder download](https://ovi.swo.moe/docs/features/multi-file-folder-download) - - [Docs - Hosting files (images) directly](https://ovi.swo.moe/docs/features/hosting-images-directly) - - [Docs - Search for files and folders](https://ovi.swo.moe/docs/features/search-for-files-and-folders) - - [Docs - Load video subtitles](https://ovi.swo.moe/docs/features/load-video-subtitles) -- [I deployed this before, how can I upgrade to the latest version?](https://ovi.swo.moe/docs/migration/updating-to-latest-version) -- [I was here before 2022, how can I migrate to the new version?](https://ovi.swo.moe/docs/migration/if-you-deployed-before-2022) -- [I got a problem during deployment ...](https://ovi.swo.moe/docs/faqs/error-on-deployment) -- I didn't find a solution / My problem is unique? [Find help in discussion forum](https://github.com/spencerwooo/onedrive-vercel-index/discussions). +## Documentation -## Server-*less* (free)? +**For more usage methods, please refer to the [Docs](https://ovi.swo.moe/docs/getting-started) written by the original author.** -Yes! Completely free with no backend server what-so-ever. (Well, we use Redis, but that's free to some extent also.) +## Modifications -## Sponsors and donations +- Compared with the original version, this version mainly extracts the steps of modifying `clientId` and `obfuscatedClientSecret` in `config/api.config.js`, as well as modifying `userPrincipalName`, `title`, and `baseDirectory` in `config/site.config.js`, and sets them as environmental variables during Vercel deployment. -Open-source is hard! If you happen to like this project and want me to keep going, please consider sponsoring me or providing a single donation! Thanks for all the love and support! +- Left `mail` in `config/site.config.js` blank (if you want to display your contact information on the page, you can modify it yourself), and removed the `GitHub` text next to the GitHub icon (because the icons on the right side of the navigation bar felt a bit too crowded). -[🧸 Please donate - 微信/支付宝](https://ovi.swo.moe/sponsor/ways) · [Patreon](https://www.patreon.com/spencerwoo) · [爱发电](https://afdian.net/@spencerwoo) +- Also added support for [Vercel Analytics](https://vercel.com/docs/concepts/analytics) to conveniently check the access situation of the shared page (needs to be enabled in the Analytics tab of the project after deployment). ## License -[MIT](LICENSE) +[MIT License](LICENSE) + +© 2021-2023 [spencer woo](https://spencerwoo.com) + +© 2023 [iRedScarf](https://github.com/iRedScarf)
- - made with ❤️ by spencer woo -
+ Made by spencer woo | Modified by iRedScarf + \ No newline at end of file diff --git a/README.zh-CN.md b/README.zh-CN.md new file mode 100644 index 0000000000..037171867b --- /dev/null +++ b/README.zh-CN.md @@ -0,0 +1,71 @@ +# OneDrive-Vercel-Index(一键部署版) + +[English](./README.md) | 简体中文 + +本项目fork自[spencerwooo/onedrive-vercel-index](https://github.com/spencerwooo/onedrive-vercel-index),基于原作者于2023年6月24日归档的版本并进行了一些小修改,让您可以一键部署在完全免费托管的Vercel,在一个网页中展示、分享、预览和下载您的OneDrive文件。具体部署方法请参考下面的说明。 + +> 本版本只测试通过E5开发者帐户,其他类型的OneDrive帐户有待进一步测试。 + +## 在线预览 + +原作者提供的[在线预览](https://drive.swo.moe) | 本一键部署版的[在线预览](https://onedrive-index-iredscarf.vercel.app) + +![demo](./public/demo.png) + +## 部署方法 + +### 前期准备 + +1. **设置OneDrive帐户的API权限:** + + 本项目是通过调用OneDrive的API来获取文件列表以及下载链接的,所以设置OneDrive帐户的API权限是必须的,获取方法请参考原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-api-权限)。 + + 需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。 + +2. **准备好在Vercel部署时填写的五个环境参数:** + +| 名称 | 描述 | 默认 | 备注 | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `null` | 例如:尼加拉瓜首富的OneDrive | +| `NEXT_PUBLIC_USER_PRINCIPLE_NAME` | 您的OneDrive帐户 | `null` | **字母大小写必须一致** | +| `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `null` | (格式为`/目录名`),根目录则填写`/` | +| `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `null` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | +| `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `null` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | + +### 部署到Vercel + +**当您做好准备工作,就可以点击下面的按钮进行部署了:** + +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) + +- 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 + +- `REDIS_URL`:如果您和我一样第一次接触这个Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好,它会自动填入项目部署后的环境变量中。 + +- `REDIS_URL`设置成功后,再重新部署一次项目。 + +**部署成功后,当您第一次访问您的`onedrive-vercel-index`页面时,会引导你进行OAuth认证(相当简单),详情请参考原作者编写的[说明文档](https://ovi.swo.moe/zh/docs/getting-started#进行认证)。** + +## 说明文档 + +**更多玩法请查阅原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/getting-started)** + +## 修改说明 + +- 本版本对比原版主要是把需要先修改`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`,以及修改`config/site.config.js`中的`userPrincipalName`、`title`和`baseDirectory`的步骤提取出来,放在Vercel部署时的环境变量设置中进行。 + +- 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 + +- 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 + +## License + +[MIT License](LICENSE) + +© 2021-2023 [spencer woo](https://spencerwoo.com) + +© 2023 [iRedScarf](https://github.com/iRedScarf) + +
+ Made by spencer woo | Modified by iRedScarf +
diff --git a/config/site.config.js b/config/site.config.js index 9a05c1ce2d..6f2f7e03b2 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -38,14 +38,14 @@ module.exports = { // [OPTIONAL] The footer component of your website. You can write HTML here, but you need to escape double // quotes - changing " to \". You can write anything here, and if you like badges, generate some with https://shields.io footer: - 'Powered by onedrive-vercel-index. Made with ❤ by SpencerWoo.', + 'Powered by onedrive-vercel-index. Made by SpencerWoo | Modified by iRedScarf | MIT License', // [OPTIONAL] This is where you specify the folders that are password protected. It is an array of paths pointing to all // the directories in which you have .password set. Check the documentation for details. protectedRoutes: [], // [OPTIONAL] Use "" here if you want to remove this email address from the nav bar. - email: 'mailto:spencer.wushangbo@gmail.com', + email: '', // [OPTIONAL] This is an array of names and links for setting your social information and links. // In the latest update, all brand icons inside font awesome is supported and the icon to render is based on the name @@ -53,7 +53,7 @@ module.exports = { links: [ { name: 'GitHub', - link: 'https://github.com/spencerwooo/onedrive-vercel-index', + link: 'https://github.com/iRedScarf/onedrive-vercel-index', }, ], From 2c35eafa206a3d1e34c8199234c3685cc339901c Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Mon, 26 Jun 2023 09:36:19 +0800 Subject: [PATCH 07/88] update README --- README.md | 2 +- README.zh-CN.md | 2 +- public/footer.png | Bin 3549 -> 0 bytes public/header.png | Bin 72302 -> 0 bytes 4 files changed, 2 insertions(+), 2 deletions(-) delete mode 100644 public/footer.png delete mode 100644 public/header.png diff --git a/README.md b/README.md index f9e5efaa64..77b396542e 100644 --- a/README.md +++ b/README.md @@ -8,7 +8,7 @@ This project is a fork from [spencerwooo/onedrive-vercel-index](https://github.c ## Demo -The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](https://onedrive-index-iredscarf.vercel.app) of this One-Click Deployment version. +The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](https://onedrive-index-demo.vercel.app) of this One-Click Deployment version. ![demo](./public/demo.png) diff --git a/README.zh-CN.md b/README.zh-CN.md index 037171867b..ec93a46315 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -8,7 +8,7 @@ ## 在线预览 -原作者提供的[在线预览](https://drive.swo.moe) | 本一键部署版的[在线预览](https://onedrive-index-iredscarf.vercel.app) +原作者提供的[在线预览](https://drive.swo.moe) | 本一键部署版的[在线预览](https://onedrive-index-demo.vercel.app) ![demo](./public/demo.png) diff --git a/public/footer.png b/public/footer.png deleted file mode 100644 index 245380a345ed7a818520c067b1d58e1251caccdb..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3549 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 S!GY}qg05V)G^sIidhl;|l3XnS diff --git a/public/header.png b/public/header.png deleted file mode 100644 index d0f2634e3021d51fd439a2c8db4ffd574889b9be..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 72302 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⪙w9>*n;Nb zY8c1mC)@Az4rGBS&U(b{{H$kev7B|EPH~Z4uRpk&E~fa}hHK-hnuP^RTM1r5PPhgT zU+wu8;dD4>$zw$Nc$_y2(^&TBt{QEiWgov~1t*>r(&Z~#^Pd8MRb9@erbK3MW;zdR)$tDwYhc%G0GKJyE3lnMEYHpa(pJ$f_^$r&{6c?O^!hC&8HAvl-%pz~^x=)0{%#8;_fxU|f77?zOs57eu zzMjowtIrp0?}>zWoR)S9{K6b8mYX3MyWG#8pJ?-(^NP#vI}Tze$o4 z8Po7URaI+k^&3|G76_HKh`5j+w4SM_n2{fO@Mzjn5RUz2#aq5arm{8Eey|wqPuhme z8M@*2mK8b(%x>ap0vGyoQbG9CCO8jE63SBHcQ4+vySl5H);{btC=iBw={5G;AYY-U+p#&d$Zd7I2G7 zR7Hyw)l_R`JfB?Rw!5xS+B$E6O?P*|S|W-v8>};UZSUcWu!CkQ8Icd?Z3SRHg62q*Y{W8J&N;@-T{?990`B+*JG12e>VQNNcr+#BWC(T4FXcP9`if@(*T|{y{ApU z%3az&>PUtYuN!CYN4`tB^4wz!neESPDcO8>_t$WHl?@WQ7b}9iAf&c86db3r z7Uk1(;bY2XnBQutFH3TJQ5w_1E`fRu-;#d5p|;<-Kh{%lCTT7@DD1wPX$45`gO9N6 z0A9aL&!=;;XE`@L<;c3VA%~VqYQSDx2Ff-is%|07C9V;28nMOLQ;xn#_0n~r~jn-@8tJ2NZLn`Wt z1d5-0dJY~ck$(N5&Oa(L1)1@^bv_QkJ(J!<%Z{v?u!;D2o+~%^Ja6-sv)J0D_3Ko< z!wJn;OEaTl51ZwykcG_J@D$M)zAJ@rEw6ja`5XEfM<2{4Ea1+NG}Q}5><*Q}peAe_ zr(K5Ce~!~0o7x4G)|RflU7f){;X8EmbIDSZp18K8-G{Xe=*p(-QGcBam5Jr1A7`-> zVCrA;;8Ee_xd-GgCJ~SPfN6&ov&y(pG(NkVKoYy54`WBX*z zKVL@pwZh>5E0z1k&Qbp=`%Qt%M$H#R%CC zCRGne3Z)1zi|pizw?fIJG3Yr9%*?oz2_h_eSPPws+v!8v zHGDR#sLWLqV>Qi@al?8@wTnAxkqCI)Qat z3`D$mML2rf$j>SLZODq|e)@&rdY_zEGP5t|Jmn*iSl&r&#OJ*9qmA*jb2b*j`w&Xs zR<<<`m@!pHim~(GxiwFtzT;=~Us|-q=#te$3Q0KYCT$_El_L7V?@BF%>gQ8$;-#jC z)n8-_t_@4pZ(_Hcv3C7y_@u)EgC=;ssVFWvSP1rz`leH&!{<7Kjt{-)v9!;V$btr7 z?*u20p^0O-6r_D0zO{%xfeQn<;I%uyP2hFdCS4;omPbBIJ@M!&?`i4NO}NDRwm{bA z_^0;2dXFt`O)J{;O5oz_BR6!cLY3X;D&n(%k%ZTq)~MU1GpxWtVhLmVo{8oF!LvaJ zV*_rQr>{oSjO!CR5qWnsan*|7`F*0@5uV?Qx1C#D&ur$z%N#3};M;725S>;IOmVH_ zA{*mzu3prfXFVa_j7^`y1 zx3fUrLkV>Lt=lIaAu_3e^IGklyC*oxR^5a-dG;GN=G5n1EG#&}`rr2)-J`>reBCq7 zXGJV*5V+A`u6s*H{g-pOQ#PB#hsqBXWcLu@Y*UBd&cdH}uqMy6meo~M1jvS+(|eLL zSJdBxTu;L8T*>2hkfzrge9ovLHmkPhA}wvYG`xT4kc_&6 z8>KcqMYapgM?8H5ZQF4oZk4tq8x-yTg}x}K(<6j5Q6wf2(IWnww$6+S^0^%EMNGn$KV4w`>!60sVR@nMnY^~TqqAo9 znXTIod*pbrzVsI`FQ5R2x|_y$?m1b-E$p6Wpu$sezfo5hr!Gywd2y zJ2Ok`IhRfp(R1LUYBaMJT}2)w8R-}IlqqOmEDIyR*p#@i}p$|ux=mMwNt z;AE2{U2pH|qZ>noB%;i_Fe7$2uA-o&qo+#$9{t15{uUPyD+5yjnG|cyBDlEX!WOVV zn;43i82V_x^U;1W@s!#Rm&hU=i9qlSjqPB==anLxE(9+5PLXOE(DO41BMRaP0`Huz zXQt15Be<-t(PfX7*jC%@{Vuhf(1%9xiHGx_lT!|P*-Qv{J{?VLu=q}1T;14Fj=1>T z;*-Ba%1w=QH*RVhRIX2t#L?~1&!e|IJmifH%V`;08>ybi~Kt7`@f(WG2Zk%1PIav7Z78D0;1EWGV zlw-r>YR&IiWur+cwv>H8v_j$D@|Pz0pWCuwSJ?lO6p4X&SL(&6`gm$@im{8}y1ym0 zK95N7cM&yM&W8{A-rSuqxYmx10@Y@+q8!o?{bU}W+s)Y6wOx;#m(k8K7ZTCD*`ae~*)3X4F zlrGIUrVoKVtD>rE@OX6Ybi&Yvy_UlWS9fRxVc2BmBx8 zLo*4#Hw*0#L)x4X?rMTz-j(%OLv9NDR_Z$LGjU0Q&K`lFebdG-#$kjh;#+HmGYx#A zx%|fxR?!$33U{wj=$BKjo3=9Z=8g5|+o!{`C_AL3*s~co&%B^aQ`FMkC!_Ys>1~rb zH5{z)pBhLD*M9PNEDyi&h~+7&YGGmW(XKAzI&j*VxA5z?$X@BQLF$Y)aMXWh0jIwh z8R^unjHzt`qWVKWd)}`o<@yWa(|-{bmGkeVry-SQ5p-P83#r%Ng?&t(?R4~huplm= zx#T~wgv4l#00!vm{%Eg5rvbV07DgO`fvB8)#*J#yg9HKRnLwpOW9So4&lLwz2}Q69 zy}g)4{9@%2j-!dQUp5bV{6u&I?Dn#1n0$y+XhE^m)#t;-5G+=oPvh+L3%E#Ihkf-< z-sE@k(^bZ!SaCG-j!kUSoh8lf(9-Sx%ZNSW96Y6vbD3MgybYq$7Mg3&7nEhfSUQzq zJz~!ouxI3I_=B!xy-CF4y$O-sDUl3)J(P$O@p>scJ_*i-nXYZpvWHZ;|qQuEVm8=eAtDTT#Rcy60v zfQr{cIj!qq7_2N&M2Azk;q z#G{ZInCA)1G0h{>ixCi1qJb~3Rjx0tdajvEe+C2T)+ROUx9;TT>lS#EK5|DPLP^-_ z-}nc|O;K;2BI*~ODDF4m7lC!h*eUw9`>Af7;<=N&Hk9}H1c~Mut!-~3^Zz9|fy=?5_28nkKtNnoMRVsT;ERi9IzU|H3Aiz|*sSH}}dm z&jDALibr)e1!t|Uw8m-!75@i8?Vq6TyNueN&mcSYXM_Y({1+6`iQ zDJm8F{|6cOmI$dLyFT8;j&%hWbT{oBy1_oqKI23M-7btE?v*jr!694niAuP2M$mG? zJj~AO>5~B@!SQ;|aALyud9mn#7dXzM-O(*a(fPFD$picFBK0J~=Rj=Sfk6n<#tW)+ ztc%*1NJJwDQt!9wtQ;HVL21{`i*W)(rw|`QB^j=KF=wDot zQN3AqHn4;3hsQ}_GZ4@m-Y68SuaP8;u`ZZUQJOW-MJIg}G4G_&*?}KIFM44-1`nP( z{wc1015W+fs93WQVRdTu0j=t5$JHDa-4TY3R6KGysVxwn2Gxi_}%nbwBRTG*!T4^mjWQDHG|TVAdjl}=Vp%*_Ct|%b+N11 z?=-J7r_0X5iJN&7L1TE$nB zM0GFq(8_$hV4UBj#a?isZ!$75-DzuAXVp$4acf zXgX))T=H1A#ReLOn%j!|hsXTKr~D+dXB) zYUKzCeN|nkDiynOfAuPEwA#qHUpvc)Y)5$QuJB267bzt4sETo;85Q>E!nQaKkx zZt6wCPZ^#h|YmwhQ#HL>$49=}V(@9vBB2iSC_*rxUeLK}{@XJOVR-C*FB902q#QFkvy z{({Iqz=}=55V({^SW+kHZ!vS=-(tqz_|?do^TZiTKZRc>ez?q4=tv5%F2z}JWP)DcO9Y7h*okdJfQwN67ql7d|Mt^<7#DTd5_u zzN)UruYSUuXKjFgLjSw#@A7CGMt*1wt5&qMF(q3@J^H<*r093lDl`I+DpMdy&II@f zGNzhiDlb^K&Auv_0#pM6Qgj#bC7gd{QDG9>yG5d%(i9B0*77LjJG87}mt?SEQS7pi zSb*>CCsLI3_Q5VWLysOr#58J)m@}=l!|@Ht53vRcE!7?rLc*?Zk6C`GHGf9$TU({6 z7SreNw~F(En>1!>COqeAS7VDA{EqIQq`q0u*2c(NB-zCL%fV~3cmhlV)UrDhboJVX zz$LQ>USF}vt8v3fORACfbz zD)xf>YYncE617e?>_s8{%BHYOb}PVMYOFpOas>dzu%5`-3Zme~Hwm~y!rc{S<502SOD||{^ayPOJV{EbqHRh4*|6*J zGpIw;?ZUbetzm`UvHNB3NWI`i(jOL)sBw(of3<6{6sB7SiEw$Bxj7s0$Ia4E*wf z>$1T5L(GU(@G_^*U*wAYkNRSFqn}Rc>B=~J*-TGF8y0T*`GeQ+>5KMg4XIGTn`~AE zfMHVs!~ELfIXq(m&zNNm^pVy!4D(3ClO>^w3-0@UH2D%Hef76pH#n3=+qHI1&}3)) zy0<14Ng7XDsA|!qSV#S3_@xS3sCTGj%bEr5sRg`y>6`YFIk|m%-DO5oPOuX9pg>Db zzC{oB-7_b3`AYN+a_%)R2<}viT7th&cQ5qfAfq%XV)-6;ObWo>^Wh5`xOl#)tyamk<$>yD-AtupclKLL?{6QYN}MEeD@X#kQ_h?SGXP zd^I!9qB`dRqqJ9b=;ffritGDCr-Y}``4VAz3a)n0qH*3LWoyw!RD&E}gDUFb51x3` z!m%;$4Ou%CGGu-lv2zc;h#3aeWrno0^icWaV`6`f?u~6~*=DO8%vY9!Ym+A*zjhk` z688HI%t%KU4#HP2GFPVKL|BB@GsC}#{b0ynW;`ON6&qyH8s(GiFalM+?yMTIvkuO+ z*xZO0H}TWY->VPoHy$uV=#3(rmJb*FHGcxa56Qrh7&s=aIJjX+?$hgeL1bRzrVs9a zum`9tYt1Shyb!X%RNia4mHWUyX2T+oXCl!F4zH%SrrBxk{M7_{O}HkyH|^Ub?RWc9 ztTZ)3n{G3#QORQnRpz~5CE7f}x#Fv(%qyEa=}1$R%ucKM5hl`n>L|CT$_q1!pLRt{ zvKhuC>&nI}n6%|%i*Ee#qvBw15J~eJR5w3usPFM_G@`2@_-rlO2sUkzOe&E$RiPJM z&#uo}@;$QuHcF^brv$#9{V|P6I_##av+yeanjII}u0IMza@n7%aR02`CJB!K4czSP zkm*_7uW>AQO7-Q{Xa?%hzaAP1bFJy`!ohUJ=j{iD`ZmdvcdX#y`AygosI%+FRMxszzBD#nOAQ^hYQ8eZXT=3={%I z;#N#B{hX2w#+Gi!zAq{yU&>VT|8$>_FE+8=yGZpQNL*Vl`L6Vb*PYD%x2B^tg(m#b6T+K2fvSDu1pB|^lptQ?xjT>zu@>b)PUM{DuKp-w|#MK@3n4jk=CS8X}Zdd zi>LLuwBf-@(}s6GNw9~Q>{YJ{3WuJ{42gt$4GnpAzMbKp4$aIMd_*5^=l}6FRmicN zBwa;Xf@~;lw#M<67NwO;b+ZZ$+C~y=Jr7A7F4fg(ezdDuL>#Rw&<h42z_(BTzHAX^?_7mKG6x>H3&;Y_ zB8%&7>t|5McivrKb%>h?D+}NA=5}**W1LM_Sd9^Y=#cxPVc2XjwfxJ+~S6d!B~ z(H>+)z;&GB!f2hkTTtj`m+NSK0M8?Am8;jiM?_b-vL!myTW=r6&2I0UluStO%LWPI ziv7vMZw8GSp*o!!`|7BwYuI;Zjw44hdwK00wNdZC3u53Z!MsRL+9xY~YL^vL#xiJ0 zJvQPw2}@#x(;F~U>h1_AH`BOFhr|M@FJI{Hj$AyG|e@l)y#cHri4W10Y#-!paNOcI-QV9|r0 zR#90Ad5Y=9wisB1>D@6K68WI^R-?)662`n+V$4viMx~=|U>pvqdsC2#gfi}6^*AmbfluJ# zD_}_pg27D{OI89ih>1U05|~)MhKk*>6m#9f4rzl80>%wH3CjF-AyZLn!G8fP^1r2@ ziZ7XY(p3E6g6e%;S99L}u5VvTj3PO+>tJ9ei=4SXnpGl!6j~}f@`ESYxx?Vs*htCO z5_xI(5`ztoiv-V2zu#{{Pn%Adn{;>!r3C`~HNo z(cZ|!^2tdNk-ltA>s_`8K-*T3uK)vCq2J&buxC`flk0wdq13@emSTNN;gFw?@$bqv(p`-Nj)^?URT|MW7+);U9}gxuy1#2VRE~`HQyW^h%qH^?k+Jc_^>^XYQ-JAZL!gHV` zEqxkGtfdQWts)tt$OvCRrb}DOFO(;j1Z^xnkT4@*c9$?P(W7UhM3r768lOeqhv)MM z^<-SD$-Hpxc$>{%w)#KfW}NSP$zpt}3&bI1k$v%Plkx40^Es~ol#!WPc>k=Qot7(f z@W`o!a)m%|Kx0Xdm|noLgubW9_gtG0_SZU)tWA~!_ zLhU%wKKABsKPMNQc_H+DOAC8OnBa)}x*YFUThp}dN z;*P0%r2?|Z?9e%(nBE&VRfPc^iII3-v7>4-To+22|A=9L4=p!$qZ;rv8yPjJNi>vw z_fs3@TWt6z3=r4K?m7Ld#FWuFQb?t|&0~=of1Zn$1Xdhc{)1u5So;vzueTY%`QI>Ue`(Zs~a2uf+mB?eO|tOGpcq@nT!BkPE&{6Ixnt zdSY8|O(qOdXQ=3Gv+FY6yY-6n;oO6|;GzfXyt|}L1MO)v82U3A$AR_=G2G)_JNi|U z#D$xfH6cvt@n57H+XA39J9J)K7xxB_(2=yvvV55C(Za#CVtpR-J#F|8T`hm~KcsiX ztwwgrTqgXLv{Q}+TkH{I1L;H&?c#yKW=~{;vhz!a{tL@qhlpkA@+`swv@pb-JP`T^ zi~7y@KYj#HuqU%h+3`|pvyN5Nfrl%8VKrJG~d&DrZPQ{?wtdXXu z_#P?-y)o&2x*EkfZ4oX)*UWQKP0}%vnOS2zk9ZMVZ0!`kVKCJacWv_sNg2D=;H})o zALfy*V$?0x5;OinC&3mHIysEqu%&N8n43s6&{#1y=;N~9z!OgzN?oz-%x!*0emjeg zh23aC9eX6wg^s>;;laniQn_e@3mD5NlNU2z=Ay^pe>FHYZYw*?4z1Pc7?Uju6-84w z`P9ze)*T`u$uBn(k@*l6PxYPdjzA4;jYiE(kFSEVe}Xs4<&I29QFfY2GlGeuq6!eV z>pgJmJ#Z?X{!Q{8L;e0#3VEXwEI~Dk4Vz%;HUO<&Qv)JJm4t5b0bIVThekuay}e0* z7af8P#AvPOPzhxR2?BKMDL48BC}d%SO0OK;uyFH&*mazecB_b=ku`Z?W-B`b^*lxC z>IEE6*eR#+F@Ac8*OA=d>jf9#!X6DjrDiL-`np}dKXSu`)6j+vPT7a152T0U;}HeJ zn;K;cn{CtxJe5JU-)iE!=rvKd(JND9i=A#tkP#ifYf)@oaT)Ki z6@0msoHx6wl4`I7#&K;-5NzrMlsyZAb^l?d=$ z>z*7vtqntIhpjK1d3mi?!1))-cc$%99B1=S6K&ajOC_Qo9SD7?GCGCJxCU#Z+&34U zcNgs$1&H+6#Jgs;t-CT$6q zk#oZ8LYM&{PlnDPH~uof9JF~3%zZy%HivdcndDk*#0q(OdisaoIi{8wnNncsmA`pn(h zHhz65?A?JHtvL3GlLFC;wWd1f&U3zn-Rpuo-ur~&Y65mc+hHe#j<0wm4nNe>m87OEegz_wV zYnR?&kwlz}{WNKuT<02kG z=~bB;n#^^Mjt=l;MCoitkY<%gyE0O4@RR)rsg>Q{RY{>un?Mtw z4U;)zqr>E6wyroe%?@rkG3Nz#NkN*M+AqW?I5>(=H=wegdg4G7=+Z&nS4OPoi7*J- z>$=+Ol-iyRprjVst}dylsj0cpD9gbi%_=TfAmW%$a>pt_Q|lmJR=Bkezf*(GdR43#l)zzRVd2f-|snB zn<)|fC+;QHuq~}^W$UR>D`G|bhVM;ZpFD2f6df9RK{ZZS$tg)RSp2@`an6sa|*=l&jBGkBn%rMCiGqy}vShA6aMZ{9@6014LS zcG0Oa6~BY~Nnm{g!N%W-czrN+zZG=OE!iT!-{=a~v84nsd{MUrzjm0u^9L>C3we{K zBlrLLL+GU+{I^|qC{HCaK1B|-s-xg&{Mgj}r2XTv5_uUJFKJ;;OE)PX?qb#c*BtCX z-~X3wU4RE@2CH1Le%k;-(|_}tXuJd!SPYoDyI|Avf@d1!N>et${pfe{pwAqwfb6w3 z#N4rGIfN{Gi&EN=Fnb=#BB@~3tz?{Y(B|Do%3g=+BP)Lp(&|+&qrZ2y5utEkw`8R~ z^~a0D%zVrSHUY=UaQ=~{=HJ#OBNgz<%<+a-hfB1ydGUfB6R6^3J3`)Bssq{cB_S(G zux$3G{BHqqCgbwN@$AwOvgz=4vo^h1n@VNw;!W^mSv*XrR?Dhwl8&Pd(KzD}U!U??4J%B?U;vkS=#}QuuUtiLd8> zwSXC}?p@FnmUEHV+ts}Q%4~|Q0`^15UvP;H5D8gpmi%Ge0mU)0B!7x=&P1s4lkyJy zzkXWHJ3`czUhk6RU|Y7`me>j8xL-0baD_y%v*U_5Uk;NMx>4^XL)<}Oa~mIn zdc+8CQvWe^jDty`?e0hdk(#V$@EiT8DA2s0g)$s3^l@8P%>`A@Q6$L}^|E6@P#kG+ zw*kDth@EWGb~ZB{JlE=0ZSx^M?TXH{RU=ua+GAY33WTf2#lla^`ih-@d;YH|{wcdIy zNT1$GtPf+|gt;E4y~eA(mQl-?OHEA`+8%2-k3AhqT-XIcq>yfZNsgoBWy@|1Zz%Qp z&2-nC9P{g?G6oAmD_Dd!9t4vhD*Q*EB?-f0J7*2H!UwKuT12RkV$ zjE=BSVIV1_ZhwEergbWw`v3W9YP<1QiV<=~rsqZM!Dv2#lWy4--ArqI*9HpVWzFy7 z_T;R+Y;<3-H9w0ko7&k}YLyS0YKnq+co=zm9H{2eMV`76;I1ZQ|EAm`XSTokxPei@ zH#o4kSg(ey9)z!wX5HirBaunYJho_;9B8k?Hyu zfmoVB?mypoPK6`NLW0ydoZy6ZIK7>j0QhXh;}Et`Ip>#CW+(A!6y;ROW`Kge>izyn zSLX*eA1F3yhN6*-BOt}lYSD~m;lAbz8(6g8EgYv-h2`O_iW&p(L`2Qc zxg%ExDJx_f11K;mc)YUh>Vt@8$|V#P$FYn7h}X%{&c)4$=MnGts@g@W^`Za?XeMGM`rJDW@om+B z>F%E7KS%S}KAtNXgH#fx?N+w=_zMxFg=uP|URpgIO{OrZIVt z%W)j$l(~%Udb?L0$=&##F{_E$PI=A8li%K(8u1(#S9F&$N7@ln5A6WvL}Gmy_-e)^ z+f2mh2$qTyFRSPp?JZ(BPUN~DVqJ!hk^fGct;Dq%Q+#}VeQQD~2DPOnB`)jk=`|;Q zoYd;n&KP*&QCk#fvzaKhz8U)$(K z?V2=d{r!2sS2FB(yW!iq@EHg_!5qyP!!Ynz(N>tJoM1-p`{f2fs|t-%^P!x2`^x8Z zRt>zM7uiqQiXAt(RMC#s%L_Sm)7~t2v)D!jj z)}X~3viK#x>ViY0pORFJuJZ6oPy5L2dT;0oo;P&t<O>b#CEnj!3 z&LMg$l#z(7&`Sied@HA;ih|XELui!y{k7F zLCR7AU$rr@d^kBfbpEX!j&rBzol2z0D@LLx-tAl(Q%i}3LH{qrp41+)_E)(<%eAh~ z?c!GxQ@p>=Rly4Qb#*mW94AE&EOvr4D^8Yh`3e|St$m{(U3$?rgxo3FZM+;VmG`L2 zF|y@=(m$Ic!&WV90*-UP&zQW>gl>9u8d$o#Sn;?1DWhCC2mR^w%(j6cPfnY~Npj^<|3K+M9# zB<&RK1Wh<$87IF&v8eD9Zwbo%Hc8NL7tG)@j#kTgX7yC}rrl{$ejB|_p_iAF9x|i8 zhC6YGv$Ss6Y8f%`yWz53#KwRUoC4YH<)e_M!qJ%ARl99XIReDY0h1;!=6Hhmk@;dB(DtNdCtmL+_)8i-h(Y^(%j=7n7|u! z$s_g(Qqb7BPb2+dD+ghd2Vsk+Ng?fW%+;gJ)!=Xznn8^BaG&O7gk8&tZ;pESNFbA_ z)+|)jaBX~Oh~4%$7vAg*pR`r>VYIE>UcG(c$L|-?QM+Es_F-*02M((X3!aUd4?Woe z1|4Z@ZCX%=Z_kcE`*xZd84_CSR zjkCO)DC^bzuqiU?9*>rGBe0|BmF;7m*z)@yD$|M zmu8oEM8p-Z|6{?zh~{M_?@cdtZ_~|`*`>u394GC~)8BkRw4IrOk~BKfwwOUPpvgXL_Eb?$WlDaRus#BiyOhY}>-ejwYKF zt6E*UqAI7b{y74Z{^qj}UyDbVhuenot~V6TyQP&MlJ9+* zhVl+TrRhH#uxT|$4|b=Vg%hrbkX3Bs>7c>m5{=pbRgCedxRkSR5sII2v25W}vXL&% z7@qmcOcJz1MDUeGtQ`+76kYcKHxe{&pXSdcr`L~}Izdx#E?`Yxc=HQ5-;8Z8e zJ14Gu!AmK>_S+CB>PwrdXvxBZDJ_NPt~1Nxahzb`j$imP>skYDil_0V0Ic8N`nWP+ zZl5~&ew_0>2Z!!S{*z@7bD)ZB$a#X9glfmkwaz@3J$-Wz407I_WX@seZxAdk=YV!c zAd@4D8HHKpOVp+~BPy62k+?zAH~_o({i=w+Qj4i6Zc_Ua`J{yiqm&fWtA8VtfBxCy z7f$h{v(Y!-Nm{h*w)v6&pZ4A}tf{T*_ug9tks?SB2nfA0O`G`KbOt>VLFhF-|_lD zsB{J6>IG;k&_FcD5fIV!I9#@}NtN~m$UFiaNFK6kRvmu*G~<#}-!6~uC(^gf5t350 zYA*nTOXSAd5N73N60?i^l%aL1^0y=&V7?sWJEcM#e?k8KlDPLrzoV(-1SI}wC}%#E zFdM!5asW7MJQ1)q^z0Rdf&cJrJK~IYf;lAlLV3v(Q$mSnI3uT6d%2urqUm+%z7y^x zW_nxODTbf*(2YQBp`hW|EXq13?8Xkq^m@q&RBM}f zXR&)owlx?h$8R9& z^zlBc{yv5C!L&g6Z4($W(^^JX_Fm8I_&^w!OSaTaI7n}CqWN^;)uF6Lu6ypCGEpAHlh(bgv@mtTp%+{=kb5 zj?;S~Mkuq(z7HOS+~P^I|3 z4=_7e$>>EAhg>6H3=`T;vmq}%2PRzEj<%=0SrT}A#>D-{N>@uww*2FX<-c8jnI~V( zKY`I#Zr4IV?l*}tD6K!22Y}~pEgx)Gk8;mxambwlvO_Otmd>J%uJJwPPSLf%L%61Szo=@kKx&l)Q?y@lf7Op9TSW9BDrwAe9kv=s?Zj@26B4ZEDxv z@u&!KBclFdzg)bjiF>)N=kdnn9#P0%SRg-X2y8GQ;`TdoHd||LlC^JhsIWkI(q&fU z1i9jYpCJ3-dRf5C+CEpx({h|p^q?WXjQprxXNxSxsUNHSrLvI^Y8%D%o`rBe*PZAz zO$Ix#A$OQRaRP>Ua6!VTYsC$HxtaMXun@2H^ECU`%aZp&Z91GK%4gr1g3Wo(*^XGZ z4B%~vNYkWnOd)X=LVLU*u(9N$8dr(OR!omsi>`_xGgf?b*s8TIzi|hd#1h4Z-dFWB z{&qfjIU}LwE;+FE<%PKypPNe1H=a?{Kyio5gd3Jz;pq0t-#U0yEgnf`lcKS>`gEQz zbErP_``CM!8)&1xk5g6hVW#>QEoEJ6r>&dDB_E%buZ(V{$p6fq6C3TVql*&Sv~Fp4 zua`fj)b0-!_9tzOHgCJb7OEeNzxHKugO@+_26S;)z@K5-jprfFPjmYD;xpuz`gl*E z<}!PmQiO|M#2!JN9ddcP`;%$BW@NJtG$eE$^#NpB-v$E9hIWC%674Q%O?IeE3mU4)>Zmp-g3yf02+Hm9ywtH#+N9>O`QZnt4WkG~?vmDi zfviDK@KD?)(t})B1!5Ku%C?dmXRdJUfZ#{6Z>fCN#Oa!cuNtNtp6YtAyA=@PPpZ2H z#l^MK4X9>K#&9bn3K|F0LdZ$-9psR#@w$S2XZHo4ppRVq!|oP)0FbfN)R)VfNbn#= zYi2?WE2XFCGPr=$?A4qZJ2?PHJPbYfC(rAZ408i3uM#FUyyM|}0!N@0$!IWWLYy<3 z_>^9^A&(EE`qh3bXO%rAfehwGeMj#5tAS7A9kQ;Bk7thfl5&{m_vP2Ivs%$mfLcqc zQe_?jDY|SuTS3ql*MCXPv0PPDCI97n_!9h?(6GW1SZ^B-`paSqsCfN@9M5t$bU!_U zMCi6X3`YAJT^5V?inUT^$@<6c{H$4vf9>J5*IVIjB1JtuewC}p zq8}MyJTeX#1vLH^hPr;)^!D>C-{Q?fA`Z2}1rFR_3^P zN=tte*{W6yGqoK=4BUT$yG!e zmfv3!pB|`el%#azy(9N$5>ceHzjKeZX5H7)OvWgId1V5EzgGSmrxrxVKr59>f7miFLO#e{s7I&CHp$^ z>J)eP0P_K%c)=)jv7r&L|M|vE3R>uShW3dd`bEXoK#De@-oES0Hj^uQh2M_jP5j>( zh6Tc=M@Zhtcv%^JO1G(m4O+>i;Js$y27cFjKFlUZ#e)fB5#zK`56<8I=I?vgS&oXrW_>~XDPh>8Rs$>1x0=E+$?4Cf ze9y*{$*MZ!6}2G zJRxu7qsZe6>$Gl=v2`Z{AeI;1ZL)tiUQ%`j^6ALmOfm?ueOL}lg>riq!drj52z*6X zFM;QAdWI{ z&Tuoe=Jo$)_(gHg#b%IJ2iuFe;#@+4rxluz193gZ*jA`+)$yka1-WOe<_mh%&sU1> zky82zQ|7k3!SgfS2C^31cVx4P>B?7`i44#!+^o9K&P?U6-I654eDX=bvHX1>ZU5{^ z>gKHeRNnb8>1jsFn7}ye7+a*?l3L6#4GqYpV$rsOB{<6k#LnKI&P*!fa8MUoU>_k48g zf6qLB>wR7(Ghv2qw1+r($GgRkg&MVLx3X*=39;_##=#Iiq9bWR*ynGT>+L5KGD?Yv zLvcGSLcbQP>wiQwq+!$CY;63N2p(v`-X8-&Scj#Ak87cR;5?^3Jvj9H#D@$c zU&BJio3?C-v%EkOcCm2*mG02?F6idc2Oy5KCetf)3FNpm@NN9~6W@GY9sWQjUR<#H zfV0P5GG$h4dIQjScox#AN zWZcZjl*YQKIh9QnjP3bx^SZ)nDqdqup zg2-mbMDyu=G&AQRkEbcg-WEUaami#!Yf>Nycdp)5dN?-A1)Z8<`Pa&uft&8}fB@+qKml`6y!~1Ai$x+3Q3$>1GrJkCpaU$JMyxB?0&& zlAf8A8*SJ}4+3(yS=GS;j@W=(NeMY-gV3|tQ`6)t(@}F#Qh~d z(YD`D%U2%jJ8CZ_ou3bMbiB@Pb2e!;-yE;@%XE-)yD|V`TsDED9bq$EwtBD`tu#|i z)tR2$X(1E(5!Pt*C(_FZjV(RP3QIR(Yk$I*F<7?xNI#35ky7e;F=Dg{?(dbEV+2R5 zif=X-4<1h}IM>3q8nb5o`HmrLJhiSs_|N=-wXvDv0v=t$2w0KFFL=ClWN=D1WI?6v z$h6Gk(@5MVr^hGnL&;|So-)A=Bfs?V;N;eorh&|@fU&uCvy!XLaQXL6Q~apVv+Q2j z*3-9z9(mTokaGTRgmrI@WbPH&f z3`^{xnmBM6Bk_9(C@@k;iV1Uc0qW9HwPa3IMFJ3(@-+Sg$D5hYvu(<+$7{kbPqZRYlOwtP%B_zrGbIamHvq5_twT_ezgM3x{QR0pDboP@Ui(w84bTMa zF*b6bIC5s-_}!WeHlB9vvwRU9yqLCSIN_sz3TugmuA8EB{$l_h85+QZo?7i%otMyI zCGOkm&K^yS)yJkS#6K?~iNnLkA{T7$kR{T{p|_@bUGsybb7AIdD(AeGGHVAmV^frT z%I37-7=`JFirl$ApMqCh1IFr|kCO~?S8bI}j|J9`HRaepc`81kb!f*cjTxb6} ze#+xzcmJ%5cza?j$Kgtc${x!7SdF(vi^>-|gIrm)wHqP3 z302LBK7ABtd3#GhE(n?6Z%Xdwc>5rKIql8pffNAci4RK?8qI0H!N{o^g^T|oS+Vj5tT?^HdLA%jfrY|bk0x$}(;st%B>=V+yr6*QZ(N%GKB9UQjpmZn9{K+!Cn=}wa@XaDmIk{au;v;8(- z2mgUn?Xcv~$St<{~EL?^{0Zg(G}a?KyqJ@>nlUuiPAiQsH!`^ zG^7J>V(gJ4c;PaAV3O*Rkd3t*kD;K+9n~Qg#j~x~S_j#6L0%_US8!*RP05GUZ*!WhEFc`MP zHHrWf3)L4xSiThkax9J`)ry6Nm1mgPYT>{xXpZH~C~LvTOYl!kmlEZ`-1SNCtJF1yo1mB8fvY zD4)?(T1|I@@iC20(i2Y>CaU`VbPOf^UD&xQI<$)z>))YPSsP!<;B94OYl%1I#M~68 z_J*%L{*D5l4B$pu(VSCP|eBl%7wssC+A!2{OU>Pw(sJErh!za^`_ssgU~ zA{aQ#$Q@~=$MCq|8sPI9alV}W4Sj&#w;O)0QUFNItB)ir_k2q4H{}{{j!}+80;)x??#2S zpK4;TgL!cu%V_AtOgT8?JpLUX~HEpY(q;5x|Wt^a*jDkNNo0<>bL)i&Dxp7=-hd4RHvjTj)TTcnUNaD`Uc zy0u5^P=gQ^_RGAfQ$NS`uRZ@d@sLejG0f2npW|%z{mx@oV6L>dVY`<`a!@{k7lp zmM;v=0}iKN27ye|w8JqWW<}B`!St(97Rd9)iHu^wIkRdZ>#z2;j()_YG2r9&ym#!$ zhb($AuXWd0Cboo8F*E&#Vs!=nPP>6u#mCB+HT9jz%9x1wVG|i75V5W!BY)&0a^+p- z|6tuj*^B=b>|3KO>%^N;wi^c2wEkgN#iw!uhmVshEDm+>qQ6pwsTQMrj@OyR`GTf+lL(KbCk9XxQ1NiJ%<1(_6}R_kkO4<8zXc8Sl7-iae2a0Mxj`X~#%qAS z_`}-?z4h*gZoSpRmnrwjWT8yVo_fN$B8ASDUj_TIkm-mdiba z$4{9&=nSiDGDbI(1j-hezRhqz3>x!=CNeA>cBW~qW?#(9wXcp0r1>tjF>Pry+B0$< zn2ykKHYMV--KFq;e>%@$&rj7t8R|F&N_Y4&9^k5qZoN;`un^O z2;u`9B_D~qL*MNCIyz1=yVFi3#gLd*Pf8m7+C{;krLdV9uH5R%=jGI;q; z;?z|!?&t0caWibwo#(3g?4QkNwXNUNERdyUgQ8jn6B?$t7Ji+atw8Z6cyOwHxS!2k zv6mkrJI^egd#%OmAMZ2-6D`xuoky87)%h&2_Q!n579bFRZGn(Ck7wL`FEP6#K&p(k~4iZ zXXh$X#$wx!1&?2+_pxM>JNke{JE3bySF9`YAj~7z`-or0a`6G)A7eR4JIU3{yk82E zQj~+=z|SE&`x%uMrIb=O!8*`JGl+{!LL})EOJL)ls207#f*M^IiGPaad3pc~1%x7= z6$>7yW(nPJ?Di9(Okr&iqBaPvDr>5@p3g{Ee1rDsLk18fSXZ7p6Bfrc`|up^1Q!)Z z>FD=u8PP1al^&G)-B$_mH0!PNUyvtC5?ls$VD(l0d_?@Kf*;&gvR|Ff0>%HMz z1_-ntlB&0YsF@VXV)eaBa z1oAzH_k<$N*od~6TGoRLtv$pM;xMOGnQOwK`LPdHGlJMjo+$u2FIb)P!(rF2x*3~mLHGK`8>qY7n$X+LDYkIFwduDGr*w93CQ1LA41gkg>vsQx%zlLf zI0m;1)TE+8XeWV~zf9a*f`F85dgN#Hl&ol)o0wiXzS$_$)=%D$AXBzgWYU&r-Z`l` zattG%tUrphN8m0Z>-jQSr=+g&DkfD2{Jk3 zc2VyWd+rb%Ff!aZm7*_M=?pr`@!u@^V)9`UIXe}UHSrmhHfFUvA*|;~iweht4t5F( z-&O?0v@Fl`!l7T|3p&GXMTi3peU5;SR;OM`Qizo*K+dU4x1 z8%SP!%)E_m0TvT;S(9gfNkS=glnMbpfoY0pEcugyd#2+l8FY((;I3SMlAE8b)Tr;m zhZD65Y5YWVtvA$+%4S^NXp&PHKuRus%id@UwYUEuze_opzaLnvoTpl4!G;aGz#`cc zh9U=v$corg;XSwrVO<()cRmwY@H99epRQLFe4c*K0_y3Q`|hT7+p=GJ)>Zp^@)_iq zw0?^IDbKcsL0CL9-R`%nJ(;4q6j>bCT>XJHQ=nL(cqe_Q-n8&?=2T;eJ*B7vs`+t@ zO1^1V3=s_<@eipvTEx`7nAZ`*wm~ue((K$^wyuKqIY{B)|d zI{wZBj?Ux~puX<^x8XD9FB09M^Qxumix z#XFwfnZ^6&sY?rIn}}=dSaVQFnMYu6dQ3r1aMQTKmPReg?hU5nJj!{-RF}DT)i1CB z=7d3IWK=7cH6yMAn~@-_WYYdjHD6BeR9!|65K!b^9z^LVKXF;xxP583RJ4uTCIxUT zoJgeyI(6qk1?NE=J`cb@tmlnPWBUYt0M7qVF7vyfx4(7X$#T)Z)`t^qLG}-2l}o+! z(`1k7)E{uw#686?n@L{O9C`3Yn;JAAY*Od6uYhz91R7`p9iKU+Uqv?P+gVz|3`_&j z&SQ4SYAr+MkMG?KaW$<2^Tb0^(BNC;HVU@FN?X|n66;T_y26NP@`WamCs)>>`N7%o zY@O`uVSQf~x^52MTJ)eO=*^Yx!)_{7z?7PC>!z4iGVBE7tGTYoIcCt)A_j{jH z2!A`}C-GkiCWNf9x`i|$*?3TOD>{;$t9s%o(I~d(C?A_oqk@Yc!amZHP0{=?tqjN zKgva218VxmgRb8@*^~`wGn*FWr6_Oy9HW!?C%={1C)!p^rtBl{VT&DH-3UVZ1F?pc zv0)*_=(}mUayxTP0CYi`{arM+;L*j8)k6pmMQole4RDuR|3~@2*>19 zW2g&Hg2OtjiM;Koh;CY(<8~y}N%*tLG#zIRoqr55D|4h2%)HrgV!oYyz-a@s zSe-`->DQy}Q%_wUv{#EWhLzA|t1^-(v&bo(Y_B=Q(#TQw;vd>f)j69$5~NaaD(Bne zA8GiRB3!+*?LczuX`d~mc@BPO3z+c!m~S>1Tp$KH0AHZ>08cq=Blt-TU$37{4gG9B zFrOxDZ^sSyqwjH_csrZs6MDuniVq5Y;1gO>Ey1A??sRvzre5!8`eiBWS2^%K*OS?4 zP*yWpZU}w^rI1C5+VIUgI1BvauVx{mX8gyQ&lhnI=`OSO7vMfG01c*sk{k3C$i6ZO z#b>N9%~t)`^svr22<$r)*uU11DR^?qCHrb8C~ZuX-ulk%%yr@W+qBrMNK8I6dCh7M zkV{5goREkKTITup7~T$N5x6VitC{_;zB8CCS6P0`hHvMPp{iVe(nMmV_xr??;5p2>29-gJsbkmJb6I_I4#Q9sR{rk z*hsbbDLY*WNkoyQ+Tb|5h@fA^y5+Zxd^UxPhhm3R0$lrkAk`XPmY`M5g#Ow3+(me? zEWr{OzrI4e)dK4H#sITm7p%NRe`%D34DuVPt!~%M+R8{3L}hy@SL=Mf@lOQNgNaS# zkZ^6FLv2p6=&PvdnP@AybSf>VCrvvCcp+soId!o-ELMsB20q27uUVPI)a);u6k$=X zY2+G%SNiURDW}s^InDt(L@()y8)M60{z?G_S9m6OdoAH>d+o%{pU(K{eeUu_Wram{8AE-OFHN;VDz&}cEpl?iWi z;zOaYvq3zVoNDjD5NF72<(i67*=7BqruoOGYw}PV=aamd&6qaKT5fa@!u3Kspg4E; zCXeItu-be8-P|FUY4!4fx|7b-eIg$lTSv}3!MpL%NosTGqYA-DjCOyPmucIlDlXDc zz!aEp2`k#9KB~_o1}pYMF+%T%(KZ)1Mf{iFatnO;{p6?Iy&$>5`u2f(VPc#MpG67!5)oU%MNB=+uP=jz&$f)$b1)(9mL=MbPHxe3A2n%KdBLmIE4mfO zYYS-3eBBs4^eJucs)QA*Kpq`;5fuxU9zQ;g`u)8RFCoCH2(l_<-EowHrk!Qdgfit^ zJtK)iC0>k(6b1rU3Xu5O{x#bR=)}cC__pKSg_TUlDvupre(aRc`f#{LPG3{boYbU$ z3b5pip3{E43xffwf&(lVp#%hUw~B_S=-Hc&(;B=u+G&c=i_S z^1OG#%w2sw{cPh*hZPmv<7WCe5154n&mhbKIkK_kwj!8eMnX7nG{I93uI;F z*jp>AGitU92Vc6w_oteVKfd0qU>!9wecpoFPrpU%Z3b&8ZixM=pb=Ji`Ecuz;6iz$ z>Z)XOuy!~8MgGAg$FJ-9`RAWrJesfFYpUy;*tA2Zdr7d6c#;yL7xNUB?@<7Ir84z+ zrp`eez=n%BpBm2osMzVlM`g_%R61)sAN<$>N1$wRh#e`wL#ZOgz>%Ja0+-8%+8yV+ z7?NJw9FmFU7Jmc>eSI$u>zg0_(s{LMVhIzzWqko>K($r!d(5QyjSM_z+MFFTFLaE- z=rq(>j+93C2nTdZL_3dcC19sVTS6x5wPJi`7g}%O_?${mk3vo)O^6T;-4u1eaA*_Zc)MX7nPU);pSTUy`CT5d6Wnk9Z zXFwI0eaA4TeJ8-T4d)d}bo7%>P+SD0&0Bzr$LkX+$uSxh&+)-ujM5d_q%xnvW9vEE zy{Pd*JuqlVcDrU^R^>EJZ%c*URFVUlZs^NfUWL7c`d&2cBKS(9ZIHlaW>vbNG+GK2 z#QWnJOc*BX%11F~?l!1}$ndvqs{_MF-2>~{B_S1lYFik4r*Fr~%rRCuWe@0^Lkhf| z4)f=T{4upm*1KVJ{w3qNnx?g+KljvU0L>(Yo-i&?69~_aNv5dUJa?A58Ig0{PjCu6 z*lU5KG53{0B9uh+pER)k`$qeg7U6$Mn?SZ#TiT($o2t{BOSh8?LJu8iCF!+0o0;i- z2a`!!#cL*#Rsqmo>Em)_cHx>v!^CFOq;X5DW47oU$VK+9ugtbfS?X~{037{5Fxg{t z_eCwxNW(NY%h!3qZCj%xOAM|*?RSyoH#u_jcus=i)yGDog8X> zi(19LRCSH8J(q%LFi*aN(0N&I&*u7%)P{O(A+ITbnftgah4lX$CGKd@)k#* zKy&1Q+q)e@BPOZ7s8Nm%{dHI+3M+Q{$jd9z*^sI9K@@UzEdGH>3w)$r)4JS4f}%>r)=8)FjBzmEn$5HB;EWOA6P5dzwbFzsR#jmQQHERqL(hn43oNX7 z*#K3os<5yv>CeWtkb(uzG)EI~&kVn=gEpOI+5dejKO5YY@^d!mns`iI_pL)uH(xxGGD=V{-44n2G( zLNs5!AuKf>BUvzfR%gVplub(W(4}=jAUkcP@3Jvqqh1n~p3^*mM%)%blqh6;3Fr=& zrYk>M+G)&Q*Io?i!_M?4F@XaVPYr&$&|DHXc4OoYL4=)jS z)L%RtFiuUQ?1FOh)*OgA9s1o}eS4d}&cWixwJvY_e-_BRxrT4`Ca%75P}$LxpwE6m z+ns^j)xUHJ3WybS(hrx0d-x$weN&}e^;_M_2?*3BXi;wNk7%}o$v}0D7iw==gF^Sk z$Lo!{!?b|bUwB~g7|Twp+tA71$mU;Xw8)bZd)I%rcxQ$DX?Pj!^Q^y-bx-Rdp4a3Z z9YK-x07Ib|-#!Ip*+Z5k!k)TNp&~U*H|ui}usZ|w$hrO@4q~<1IIiy9;$bed{b*fz!nywnJk^ixsW#Y+}I`xa%{yD|O>fsU+!3O4_~vwuAo(6CTi@{cgW=)dZ%P%-y> zThgyc8vYRC^mgI4FmuXRhJDMUV5Jcv7H}7VJuk|_Foro(#6;QEt7g+7bCiJ0ho{;| zd=l9V+Fq-v+%OEB_vlJr^>AZcZlt;WwUnjN8(`06E5z)KLKaBig4KnEEw3C->otWno*IC#dy>m^ZhN3%GKl}X@N*^NE*3)`+ z8e#2%9YiiY4cnW;Z!g?ex9`d`^j2>u+ZT81@6A1yJ-_`Ec>X1zJz5cy-IU~tLOcHV zQTnHu`MM*x^j(^K1G~sW1gOVRXm@ie5+URH++Qp&N@2O13^2Jx!vtp0X1Cvh+S_kY z2=_Q#Il1$!hgJrel(ar<{vKRCaY8>b{1P)eU5d)6^(Z${&8*zh&5#hSTgJ`!+GQ%P z83`<>S@n{{ym7wF27suNlt;nK$8mJ9xXt=n8Ji00tN9t9blDDj3J|cjf&MbDzt8bF z0%U-RXZ0UK-RpGb$%~(@d)UD<>@@PhNw!Ph{yC#T%^>akz^O&g=ED$d^C*qrvSggd zYFxusT|}&)wLB@Z{?vPQrDDpvzRm>yA~Bu0P$Z}hOj;SOpS8rm+q0j4+VN#5u-T`M zHb=F3K|ssGaLS2jL4@Kss80S3NbAA;oyqv-Uv#QHB{!tBw)FoL!vhpOkNetR1}9n8 zZP{Jr4da{MA{h>^hWWY!1y)Nrm)}C@%6$Mtt@gn8&|RRJ!&mUmQ>j*t>loA(DtOcE z@;v;i1QXzAKJ7H8&Y_W=`DmnPclT)HTa}p4sTg`$XuL-1hYb3SWPLxjasj}~kv3jY zK;U}e3~NGixo|siTSSTD^mZ%TGiqvR+X18%N71y~+O4>LYy*T_r&3*4X9@fx2viPkgA|d80pxmBNT1}kZdh=P@Wo5wjJZSG z%dF2K<+PaP8_Twc7>dZ0eJOQ;gs-@_o>#of7Zwh|q zx<*v))Plyko%(!=mdbNSfHL+AuE zoAj$`oNu1YZ&fWnQSn+w>~Uxt^_y^YxtD@J07t0U!%B{lcwm9JTZk9f z#lVrvZe9WTY<2_vZa7Z|)#k?zAU^F-rFG$r7smGsoF#n?3wXcgWDBoKKHjxIZ&@490_shOqD`K>chq~uJh-l+mVf|gl>ewf(xg&#?`B`U9G-nRx z^!))Z<+VObe#7Br2LEB|MQ1Z@MO=%a;OX&k#Lh}akh(+$=V%jb=lxbiY@q#bkm9FS9GtZ;0J!MH+QP{ASyRo3<4 zuZJ`7Ol0|25#P+y9!^j=YMYqZ1a?rmDmYIYkjZgTerF-?jyPtZtfd z9vS(iiXA7Qx&S~K>2kt|zX(z$B#F{>e8w*=1HciLjR4tsu+`t17?VMA2i&!W^5KBF zU9~ER73qtYL`lsq^pYGPFQz&!{LKFrW~9SpP~Z=aZs6!(QC3WW7x1=)>sFM6?B8nn zqQR?>m*4fKn=vmHOjA=JzYOQ%VASq)fHflTPvke++;(W=|HcwQdUH9b#^Qf`=}oah z=;_j0Fa7G@0$tCF!id~DHT1IKycS&TN zXgj8ctlLw#roa(+K|%H8l)P762M6w=Zm;88Je>7g%a?NNkoF(fewnP#%Y?tVMuBX1 z461MqXY{g1H$smTNTE(nla*zjG<5?I1v}6zO`R)5R6<6t0ch&ZBaCzbw3utUPT1L1AvAJTga!- zu>Gbi{wAEh2Bt%$#Z?k?JhjGsX$Y|rE43UsiDwEQ9u;_apWJA@#E$a;q#Qv@?U?gm ztw)xMLQP^F7hK`a$b%ARCj7a6EV|wNU0~AsRj||B$QjbfKI}_s$jNr4X|$6aHC*Qq zvMI~OHKcBY&7L&qC@-DMR*I~kI_Rd|Nxo9-YafHev@SKtCJ{HtvSD9pzl~GL?H!&Z>Ewr3Yq2 z3F4%@T%63XzNs7%#BiocDd|g+3#InK?oQ0FeUf<(PORM*^n3FJiT;x=ei=eMGHb!U z5;t8o>^t4(jU8B28zRe(^MOXe%tF`y8%G;^!fMV^XKsMzbYgC z9FzWOn)BySLg0V)P7S>;DcM{f`st-xSjS z@FpX2U;URr^v?)=zWzr5@aOpd=yCgBKCAOL&CmBDj;If^eL~jyj|e*dt0m<>FEIb* z(SKic{+#CGtKS=lKZpPPtLpC0F=Ohk3!ar)k)kB~cxT7IruV;mqyIGL|Kpqc4bgvS zq<{X8uf=b*`2Xgw(*90o|8ITy&xiis%m2+IK;Y@av-b*MHFxB+t<9tSgd}?e{(qmo ze=kFS9!UxKau(}<7@dD^-uS)p|9Ow`|KxG||ICp8@yku5h?6CXjRz}|n9GrbjXTi) wiWB!=!sGt)o8Ni75WjVke@^p%X(X Date: Mon, 26 Jun 2023 10:03:34 +0800 Subject: [PATCH 08/88] Update site.config.js --- config/site.config.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/site.config.js b/config/site.config.js index 6f2f7e03b2..b66cac4f6d 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -38,7 +38,7 @@ module.exports = { // [OPTIONAL] The footer component of your website. You can write HTML here, but you need to escape double // quotes - changing " to \". You can write anything here, and if you like badges, generate some with https://shields.io footer: - 'Powered by onedrive-vercel-index. Made by SpencerWoo | Modified by iRedScarf | MIT License', + 'Powered by onedrive-vercel-index | Made by SpencerWoo | Modified by iRedScarf | MIT License', // [OPTIONAL] This is where you specify the folders that are password protected. It is an array of paths pointing to all // the directories in which you have .password set. Check the documentation for details. From 2350d03f07ac6d2e75cc955ec8c5ab7e03436007 Mon Sep 17 00:00:00 2001 From: eks Date: Thu, 29 Jun 2023 10:01:57 +0800 Subject: [PATCH 09/88] update Test environment variable key name --- README.md | 2 +- README.zh-CN.md | 10 ++++++++-- config/site.config.js | 6 +++--- 3 files changed, 12 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 77b396542e..215e9df29c 100644 --- a/README.md +++ b/README.md @@ -8,7 +8,7 @@ This project is a fork from [spencerwooo/onedrive-vercel-index](https://github.c ## Demo -The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](https://onedrive-index-demo.vercel.app) of this One-Click Deployment version. +The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](https://odi-demo.freeloop.one) of this One-Click Deployment version. ![demo](./public/demo.png) diff --git a/README.zh-CN.md b/README.zh-CN.md index ec93a46315..84e5b183ba 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -8,7 +8,7 @@ ## 在线预览 -原作者提供的[在线预览](https://drive.swo.moe) | 本一键部署版的[在线预览](https://onedrive-index-demo.vercel.app) +原作者提供的[在线预览](https://drive.swo.moe) | 本一键部署版的[在线预览](https://odi-demo.freeloop.one) ![demo](./public/demo.png) @@ -36,7 +36,7 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 @@ -58,6 +58,12 @@ - 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 +## 安全风险和一些问题 + +- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。应该避免在以`NEXT_PUBLIC_`开头的环境变量中存储敏感信息,如API密钥或数据库密码。这些信息应该只在服务器端代码中使用,并且应该使用不带`NEXT_PUBLIC_`前缀的环境变量来存储。 + +- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`obfuscatedClientSecret`在没有OneDrive帐户的登录密码时,也不算是太敏感的信息,就暂时这样解决了。 + ## License [MIT License](LICENSE) diff --git a/config/site.config.js b/config/site.config.js index b66cac4f6d..0f7afa5ae6 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -7,7 +7,7 @@ module.exports = { // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about // your email being exposed in public. - userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPLE_NAME, + userPrincipalName: process.env.USER_PRINCIPLE_NAME, // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. @@ -17,10 +17,10 @@ module.exports = { kvPrefix: process.env.KV_PREFIX || '', // The name of your website. Present alongside your icon. - title: process.env.NEXT_PUBLIC_SITE_TITLE || 'OneDrive-Vercel-Index', + title: process.env.SITE_TITLE || 'OneDrive-Vercel-Index', // The folder that you are to share publicly with onedrive-vercel-index. Use '/' if you want to share your root folder. - baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', + baseDirectory: process.env.BASE_DIRECTORY || '/', // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. From f3f570427e68dd14bf4a88698d3a287648a1824d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Thu, 29 Jun 2023 11:45:21 +0800 Subject: [PATCH 10/88] Update site.config.js --- config/site.config.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/site.config.js b/config/site.config.js index 0f7afa5ae6..a43fcfdab7 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -7,7 +7,7 @@ module.exports = { // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about // your email being exposed in public. - userPrincipalName: process.env.USER_PRINCIPLE_NAME, + userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPLE_NAME, // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. From 80c049ba0d3d8f0024c43c6448f06682df490203 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Thu, 29 Jun 2023 11:46:35 +0800 Subject: [PATCH 11/88] Update site.config.js --- config/site.config.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/config/site.config.js b/config/site.config.js index a43fcfdab7..b66cac4f6d 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -17,10 +17,10 @@ module.exports = { kvPrefix: process.env.KV_PREFIX || '', // The name of your website. Present alongside your icon. - title: process.env.SITE_TITLE || 'OneDrive-Vercel-Index', + title: process.env.NEXT_PUBLIC_SITE_TITLE || 'OneDrive-Vercel-Index', // The folder that you are to share publicly with onedrive-vercel-index. Use '/' if you want to share your root folder. - baseDirectory: process.env.BASE_DIRECTORY || '/', + baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. From a046a0f42518743ebb584b34145a9b985024449e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Thu, 29 Jun 2023 12:07:57 +0800 Subject: [PATCH 12/88] Update README.zh-CN.md --- README.zh-CN.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 84e5b183ba..2686926ff6 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,7 +36,7 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 From 5ae4d8ce371c969babd0088cff2eb0dc5942d9f7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Thu, 29 Jun 2023 12:43:36 +0800 Subject: [PATCH 13/88] =?UTF-8?q?=E6=9B=B4=E6=96=B0=20README.zh-CN.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.zh-CN.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 2686926ff6..1a0cff0c4a 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -62,7 +62,7 @@ - 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。应该避免在以`NEXT_PUBLIC_`开头的环境变量中存储敏感信息,如API密钥或数据库密码。这些信息应该只在服务器端代码中使用,并且应该使用不带`NEXT_PUBLIC_`前缀的环境变量来存储。 -- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`obfuscatedClientSecret`在没有OneDrive帐户的登录密码时,也不算是太敏感的信息,就暂时这样解决了。 +- 最开始在把`config/api.config.js`和`config/site.config.js`中的一些参数放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,也会在OAuth认证第三步时无法获取`USER_PRINCIPLE_NAME`而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`都不是环境变量的键值设置。为了顺利部署,只好所有参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 ## License From ad7dd54531fe0d9d08ca09abc704e585b03c402d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Thu, 29 Jun 2023 16:02:57 +0800 Subject: [PATCH 14/88] Update README.zh-CN.md --- README.zh-CN.md | 16 ++++++++++++++-- 1 file changed, 14 insertions(+), 2 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 1a0cff0c4a..ed8fbaa138 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -58,11 +58,23 @@ - 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 -## 安全风险和一些问题 +## 安全风险 + +- 当前版本会在网页的源代码中泄露部署者的OneDrive帐号`USER_PRINCIPLE_NAME`,亦可在OAuth认证第一步的网址中查看到部署者的`clientId`和`obfuscatedClientSecret`,这两个问题在原作者的归档版本同样存在。 - 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。应该避免在以`NEXT_PUBLIC_`开头的环境变量中存储敏感信息,如API密钥或数据库密码。这些信息应该只在服务器端代码中使用,并且应该使用不带`NEXT_PUBLIC_`前缀的环境变量来存储。 -- 最开始在把`config/api.config.js`和`config/site.config.js`中的一些参数放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,也会在OAuth认证第三步时无法获取`USER_PRINCIPLE_NAME`而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`都不是环境变量的键值设置。为了顺利部署,只好所有参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 +- 在最开始把`config/api.config.js`和`config/site.config.js`中的一些参数放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,也会在OAuth认证第三步时无法获取`USER_PRINCIPLE_NAME`而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 + +## 待办备忘 + +- 把`config/site.config.js`中的加密文件夹`protectedRoutes`参数放在环境变量中设置。 + +- 把密码放在环境变量而非`.password`文件。 + +- 深入研究原版本代码,争取不以`NEXT_PUBLIC_`开头的环境变量键名实现功能。 + +- 重新设计LOGO,原LOGO对比度太低,与页面中其他图标和字体风格不够一致。 ## License From 907267636a56234a3265ddc33c054359cac901cb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Thu, 29 Jun 2023 16:20:32 +0800 Subject: [PATCH 15/88] Update README.md --- README.md | 20 +++++++++++++++++++- 1 file changed, 19 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 215e9df29c..93bd0f53ab 100644 --- a/README.md +++ b/README.md @@ -58,6 +58,24 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - Also added support for [Vercel Analytics](https://vercel.com/docs/concepts/analytics) to conveniently check the access situation of the shared page (needs to be enabled in the Analytics tab of the project after deployment). +## Security Risks + +- The current version reveals the OneDrive account `USER_PRINCIPLE_NAME` of the deployer in the webpage source code. The `clientId` and `obfuscatedClientSecret` of the deployer can also be seen in the URL of the first step of OAuth authentication. These issues also exist in the archived version of the original author. + +- Due to the design decision of Next.js, environment variables starting with `NEXT_PUBLIC_` are available not only on the server side but also on the client side (browser). This means that any environment variables starting with `NEXT_PUBLIC_` will be included in the built JavaScript files and will be sent to the user's browser. Therefore, anyone who visits your website can view the values of these environment variables by examining the source code of the website or network requests. It's clear that one should avoid storing sensitive information, such as API keys or database passwords, in environment variables starting with `NEXT_PUBLIC_`. This information should only be used in server-side code, and should be stored in environment variables without the `NEXT_PUBLIC_` prefix. + +- At the beginning, when setting some parameters in `config/api.config.js` and `config/site.config.js` in the environment variables, I tried to use environment variable key names that do not start with `NEXT_PUBLIC_`. However, in the first step of OAuth authentication, it couldn't get the values of `clientId` and `obfuscatedClientSecret`, and couldn't pass the authentication in the third step of OAuth authentication because it couldn't get `USER_PRINCIPLE_NAME`. Including `SITE_TITLE` and `BASE_DIRECTORY` are not set as environment variable key values. In order to deploy smoothly, I had to temporarily use environment variable key names that start with `NEXT_PUBLIC_`. + +## Todo List + +- Set the `protectedRoutes` parameter of encrypted folders in `config/site.config.js` in the environment variables. + +- Put the password in the environment variables instead of the `.password` file. + +- Deepen the study of the original version of the code and strive to implement the function with environment variable key names that do not start with `NEXT_PUBLIC_`. + +- Redesign the LOGO. The contrast of the original LOGO is too low, and it is not consistent enough with the style of other icons and fonts on the page. + ## License [MIT License](LICENSE) @@ -68,4 +86,4 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h
Made by spencer woo | Modified by iRedScarf -
\ No newline at end of file + From a6628172e0477720b26d6a451d924fa107399688 Mon Sep 17 00:00:00 2001 From: eks Date: Fri, 30 Jun 2023 09:51:09 +0800 Subject: [PATCH 16/88] Update step-1.tsx Hide the values of clientId and clientSecret displayed in the first step of OAuth authentication, and only display the first 6 characters and the last 6 characters. --- src/pages/onedrive-vercel-index-oauth/step-1.tsx | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) mode change 100644 => 100755 src/pages/onedrive-vercel-index-oauth/step-1.tsx diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx old mode 100644 new mode 100755 index 255478d96f..b289fd5289 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -10,6 +10,15 @@ import Navbar from '../../components/Navbar' import Footer from '../../components/Footer' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' +function obfuscateSensitiveData(data) { + if (data.length <= 12) { + return data; + } + const start = data.substring(0, 6); + const end = data.substring(data.length - 6); + return `${start}******${end}`; +} + export default function OAuthStep1() { const router = useRouter() @@ -73,7 +82,7 @@ export default function OAuthStep1() { CLIENT_ID - {apiConfig.clientId} + {obfuscateSensitiveData(apiConfig.clientId)} @@ -81,7 +90,7 @@ export default function OAuthStep1() { CLIENT_SECRET* - {apiConfig.obfuscatedClientSecret} + {obfuscateSensitiveData(apiConfig.obfuscatedClientSecret)} From d28b7f87a42840fa0dc49f3963e990301f2f92dc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Fri, 30 Jun 2023 10:16:06 +0800 Subject: [PATCH 17/88] Update README.zh-CN.md --- README.zh-CN.md | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index ed8fbaa138..f7b93d4acd 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -54,17 +54,19 @@ - 本版本对比原版主要是把需要先修改`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`,以及修改`config/site.config.js`中的`userPrincipalName`、`title`和`baseDirectory`的步骤提取出来,放在Vercel部署时的环境变量设置中进行。 +- 在进行OAuth认证第一步的页面中隐藏了`clientId`和`obfuscatedClientSecret`的具体值,只显示值的前6位和后6位用于核对。 + - 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 - 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 ## 安全风险 -- 当前版本会在网页的源代码中泄露部署者的OneDrive帐号`USER_PRINCIPLE_NAME`,亦可在OAuth认证第一步的网址中查看到部署者的`clientId`和`obfuscatedClientSecret`,这两个问题在原作者的归档版本同样存在。 +- 当前版本会在网页的源代码中泄露部署者的OneDrive帐号`USER_PRINCIPLE_NAME`,亦可在OAuth认证第二步用来获取授权码的链接中查看到部署者的`clientId`,`obfuscatedClientSecret`则可在OAuth认证第一步的源代码中查看到。这些问题在原作者的归档版本同样存在。 - 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。应该避免在以`NEXT_PUBLIC_`开头的环境变量中存储敏感信息,如API密钥或数据库密码。这些信息应该只在服务器端代码中使用,并且应该使用不带`NEXT_PUBLIC_`前缀的环境变量来存储。 -- 在最开始把`config/api.config.js`和`config/site.config.js`中的一些参数放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,也会在OAuth认证第三步时无法获取`USER_PRINCIPLE_NAME`而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 +- 在最开始把`config/api.config.js`和`config/site.config.js`中的一些参数放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,无法获取到`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 ## 待办备忘 @@ -72,7 +74,9 @@ - 把密码放在环境变量而非`.password`文件。 -- 深入研究原版本代码,争取不以`NEXT_PUBLIC_`开头的环境变量键名实现功能。 +- 深入研究原版本代码,争取不以`NEXT_PUBLIC_`开头的环境变量键名实现功能,以提高安全性。 + +- 在完成OAuth认证后,关闭OAuth认证通道,或优化OAuth认证流程,争取不对外泄露`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值。 - 重新设计LOGO,原LOGO对比度太低,与页面中其他图标和字体风格不够一致。 From 3cfd03d97bc1682c25da70b42c599c1a491f2ed8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Fri, 30 Jun 2023 10:30:46 +0800 Subject: [PATCH 18/88] Update README.md --- README.md | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 93bd0f53ab..5252c0d40f 100644 --- a/README.md +++ b/README.md @@ -52,15 +52,17 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ## Modifications -- Compared with the original version, this version mainly extracts the steps of modifying `clientId` and `obfuscatedClientSecret` in `config/api.config.js`, as well as modifying `userPrincipalName`, `title`, and `baseDirectory` in `config/site.config.js`, and sets them as environmental variables during Vercel deployment. +- Compared with the original version, this version mainly extracts the steps of modifying `clientId` and `obfuscatedClientSecret` in `config/api.config.js`, as well as modifying `userPrincipalName`, `title`, and `baseDirectory` in `config/site.config.js`, and now sets them as environmental variables during Vercel deployment. -- Left `mail` in `config/site.config.js` blank (if you want to display your contact information on the page, you can modify it yourself), and removed the `GitHub` text next to the GitHub icon (because the icons on the right side of the navigation bar felt a bit too crowded). +- The specific values of `clientId` and `obfuscatedClientSecret` are hidden on the page of the first step of OAuth authentication and only the first 6 and last 6 characters are displayed for verification. + +- The mail field in config/site.config.js has been left blank, you can modify this yourself if you want to display your contact information on the page (the demo of this version shows an Email icon). Additionally, the word GitHub next to the GitHub icon has been removed (because the icons on the right side of the navigation bar felt a bit too crowded). - Also added support for [Vercel Analytics](https://vercel.com/docs/concepts/analytics) to conveniently check the access situation of the shared page (needs to be enabled in the Analytics tab of the project after deployment). ## Security Risks -- The current version reveals the OneDrive account `USER_PRINCIPLE_NAME` of the deployer in the webpage source code. The `clientId` and `obfuscatedClientSecret` of the deployer can also be seen in the URL of the first step of OAuth authentication. These issues also exist in the archived version of the original author. +- The current version leaks the deployer's OneDrive account `USER_PRINCIPLE_NAME` in the source code of the web page, and the deployer's `clientId` can be seen in the link used to obtain the authorization code in the second step of OAuth authentication. `obfuscatedClientSecret` can be seen in the source code of the first step of OAuth authentication. These problems also exist in the archived version of the original author. - Due to the design decision of Next.js, environment variables starting with `NEXT_PUBLIC_` are available not only on the server side but also on the client side (browser). This means that any environment variables starting with `NEXT_PUBLIC_` will be included in the built JavaScript files and will be sent to the user's browser. Therefore, anyone who visits your website can view the values of these environment variables by examining the source code of the website or network requests. It's clear that one should avoid storing sensitive information, such as API keys or database passwords, in environment variables starting with `NEXT_PUBLIC_`. This information should only be used in server-side code, and should be stored in environment variables without the `NEXT_PUBLIC_` prefix. @@ -72,7 +74,9 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - Put the password in the environment variables instead of the `.password` file. -- Deepen the study of the original version of the code and strive to implement the function with environment variable key names that do not start with `NEXT_PUBLIC_`. +- Deepen the study of the original version of the code and strive to implement the function with environment variable key names that do not start with `NEXT_PUBLIC_`, to improve security. + +- After completing OAuth authentication, close the OAuth authentication channel, or optimize the OAuth authentication process, and strive not to leak the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret`. - Redesign the LOGO. The contrast of the original LOGO is too low, and it is not consistent enough with the style of other icons and fonts on the page. From 727be2587f32b6a2be0e197cc54dad2184fd21c2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Fri, 30 Jun 2023 10:32:02 +0800 Subject: [PATCH 19/88] Update README.zh-CN.md --- README.zh-CN.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index f7b93d4acd..b2b4bd3c5d 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -56,7 +56,7 @@ - 在进行OAuth认证第一步的页面中隐藏了`clientId`和`obfuscatedClientSecret`的具体值,只显示值的前6位和后6位用于核对。 -- 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 +- 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改,本版本的Demo显示了Email图标),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 - 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 From bae3360c30ce378934a83fabccbee53be9fe85f4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Fri, 30 Jun 2023 12:00:37 +0800 Subject: [PATCH 20/88] Update README.md --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 5252c0d40f..58fd29d48b 100644 --- a/README.md +++ b/README.md @@ -70,13 +70,13 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ## Todo List -- Set the `protectedRoutes` parameter of encrypted folders in `config/site.config.js` in the environment variables. +- Transition the `protectedRoutes` settings from `config/site.config.js` to environment variables. - Put the password in the environment variables instead of the `.password` file. - Deepen the study of the original version of the code and strive to implement the function with environment variable key names that do not start with `NEXT_PUBLIC_`, to improve security. -- After completing OAuth authentication, close the OAuth authentication channel, or optimize the OAuth authentication process, and strive not to leak the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret`. +- Close the OAuth authentication pathway after completing the OAuth authentication, or optimize the OAuth authentication process, and strive not to leak the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret`. - Redesign the LOGO. The contrast of the original LOGO is too low, and it is not consistent enough with the style of other icons and fonts on the page. From 2b33726c3d8f055848b2baaae7d3ac461fe96331 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 1 Jul 2023 00:44:16 +0800 Subject: [PATCH 21/88] =?UTF-8?q?=E6=9B=B4=E6=96=B0=20step-1.tsx?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 使用getAccessToken函数验证是否已通过OAuth认证,若是则返回主页,否则进行正常进行OAuth认证。 --- src/pages/onedrive-vercel-index-oauth/step-1.tsx | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx index b289fd5289..3eb78cdd86 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -9,6 +9,7 @@ import apiConfig from '../../../config/api.config' import Navbar from '../../components/Navbar' import Footer from '../../components/Footer' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' +import { getAccessToken } from '../api' // 导入getAccessToken函数 function obfuscateSensitiveData(data) { if (data.length <= 12) { @@ -157,6 +158,17 @@ export default function OAuthStep1() { } export async function getServerSideProps({ locale }) { + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + // 如果访问令牌不存在,正常渲染页面 return { props: { ...(await serverSideTranslations(locale, ['common'])), From 48b310fa0e6a4f292de9e34f22992fc1db6a63de Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 1 Jul 2023 00:45:14 +0800 Subject: [PATCH 22/88] =?UTF-8?q?=E6=9B=B4=E6=96=B0=20step-2.tsx?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/pages/onedrive-vercel-index-oauth/step-2.tsx | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/src/pages/onedrive-vercel-index-oauth/step-2.tsx b/src/pages/onedrive-vercel-index-oauth/step-2.tsx index f4f912ce2f..dd4fd70e00 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-2.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-2.tsx @@ -11,6 +11,7 @@ import Navbar from '../../components/Navbar' import Footer from '../../components/Footer' import { LoadingIcon } from '../../components/Loading' import { extractAuthCodeFromRedirected, generateAuthorisationUrl } from '../../utils/oAuthHandler' +import { getAccessToken } from '../api' // 导入getAccessToken函数 export default function OAuthStep2() { const router = useRouter() @@ -142,6 +143,17 @@ export default function OAuthStep2() { } export async function getServerSideProps({ locale }) { + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + // 如果访问令牌不存在,正常渲染页面 return { props: { ...(await serverSideTranslations(locale, ['common'])), From 9131979263f0e11d3722ba58cbbf7262ea730d0d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 1 Jul 2023 01:11:20 +0800 Subject: [PATCH 23/88] =?UTF-8?q?=E6=9B=B4=E6=96=B0=20README.zh-CN.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.zh-CN.md | 16 ++++++++++++---- 1 file changed, 12 insertions(+), 4 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index b2b4bd3c5d..ca36cc1771 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -56,17 +56,25 @@ - 在进行OAuth认证第一步的页面中隐藏了`clientId`和`obfuscatedClientSecret`的具体值,只显示值的前6位和后6位用于核对。 +- 在OAuth认证的第一步和第二步页面代码中,添加了检查是否已通过OAuth认证的逻辑,若网站已完成OAuth认证,则重新定向到首页,否则正常进行认证流程。 + - 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改,本版本的Demo显示了Email图标),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 - 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 ## 安全风险 -- 当前版本会在网页的源代码中泄露部署者的OneDrive帐号`USER_PRINCIPLE_NAME`,亦可在OAuth认证第二步用来获取授权码的链接中查看到部署者的`clientId`,`obfuscatedClientSecret`则可在OAuth认证第一步的源代码中查看到。这些问题在原作者的归档版本同样存在。 +- 在本版本和原作者的归档版本中,都会在网页的源代码中泄露部署者的OneDrive帐号`USER_PRINCIPLE_NAME`。 + +- 原作者的归档版本中,可在OAuth认证第二步用来获取授权码的链接中查看到部署者的`clientId`,`obfuscatedClientSecret`则可在OAuth认证第一步的源代码中查看到。 + +> 本版本初步尝试在进行OAuth认证流程时检查是否已通过认证,若已通过则重定向至首页,否则才进行OAuth认证流程,设想如此来阻止有心之人通过OAuth认证的链接地址来获取`clientId`和`obfuscatedClientSecret`的值。 -- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。应该避免在以`NEXT_PUBLIC_`开头的环境变量中存储敏感信息,如API密钥或数据库密码。这些信息应该只在服务器端代码中使用,并且应该使用不带`NEXT_PUBLIC_`前缀的环境变量来存储。 +- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 -- 在最开始把`config/api.config.js`和`config/site.config.js`中的一些参数放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,无法获取到`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 +> 本版本使用的环境变量均以`NEXT_PUBLIC_`开头(否则无法正常运行)…… +> +> 在最开始有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,无法获取到`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 ## 待办备忘 @@ -76,7 +84,7 @@ - 深入研究原版本代码,争取不以`NEXT_PUBLIC_`开头的环境变量键名实现功能,以提高安全性。 -- 在完成OAuth认证后,关闭OAuth认证通道,或优化OAuth认证流程,争取不对外泄露`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值。 +- 在完成OAuth认证后,关闭OAuth认证通道(目前已初步完成,但相关安全风险有无解除还有待进一步检验),争取不泄露`USER_PRINCIPLE_NAME`~~、`clientId`和`obfuscatedClientSecret`~~的值。 - 重新设计LOGO,原LOGO对比度太低,与页面中其他图标和字体风格不够一致。 From f712c006d583d3b2479a6fbb77f5b3ba9ec6b1f5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 1 Jul 2023 01:36:03 +0800 Subject: [PATCH 24/88] =?UTF-8?q?=E6=9B=B4=E6=96=B0=20README.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 16 ++++++++++++---- 1 file changed, 12 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 58fd29d48b..f1e24b3508 100644 --- a/README.md +++ b/README.md @@ -62,11 +62,17 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ## Security Risks -- The current version leaks the deployer's OneDrive account `USER_PRINCIPLE_NAME` in the source code of the web page, and the deployer's `clientId` can be seen in the link used to obtain the authorization code in the second step of OAuth authentication. `obfuscatedClientSecret` can be seen in the source code of the first step of OAuth authentication. These problems also exist in the archived version of the original author. +- In both this version and the original archived version, the deployer's OneDrive account `USER_PRINCIPLE_NAME` is leaked in the source code of the webpage. -- Due to the design decision of Next.js, environment variables starting with `NEXT_PUBLIC_` are available not only on the server side but also on the client side (browser). This means that any environment variables starting with `NEXT_PUBLIC_` will be included in the built JavaScript files and will be sent to the user's browser. Therefore, anyone who visits your website can view the values of these environment variables by examining the source code of the website or network requests. It's clear that one should avoid storing sensitive information, such as API keys or database passwords, in environment variables starting with `NEXT_PUBLIC_`. This information should only be used in server-side code, and should be stored in environment variables without the `NEXT_PUBLIC_` prefix. +- In the original archived version, the deployer's `clientId` can be seen in the link used to obtain the authorization code in OAuth Step-2. The `obfuscatedClientSecret` can be seen in the source code of the OAuth Step-1. -- At the beginning, when setting some parameters in `config/api.config.js` and `config/site.config.js` in the environment variables, I tried to use environment variable key names that do not start with `NEXT_PUBLIC_`. However, in the first step of OAuth authentication, it couldn't get the values of `clientId` and `obfuscatedClientSecret`, and couldn't pass the authentication in the third step of OAuth authentication because it couldn't get `USER_PRINCIPLE_NAME`. Including `SITE_TITLE` and `BASE_DIRECTORY` are not set as environment variable key values. In order to deploy smoothly, I had to temporarily use environment variable key names that start with `NEXT_PUBLIC_`. +> This version checks whether authentication has already been passed when performing the OAuth authentication process. If it has, it redirects to the homepage; otherwise, it proceeds with the OAuth authentication process. It attempts to prevent individuals with malicious intent from obtaining the values of `clientId` and `obfuscatedClientSecret` through the link address of OAuth authentication. + +- Due to the design decisions of Next.js, environment variables starting with `NEXT_PUBLIC_` are available not only on the server side but also on the client side (browser). This means that any environment variable starting with `NEXT_PUBLIC_` will be included in the built JavaScript file and sent to the user's browser. Therefore, anyone visiting your website can see the values of these environment variables by looking at the source code of the website or network requests. + +> All environment variables used in this version start with `NEXT_PUBLIC_` (otherwise it cannot function properly)... +> +> Initially, there were attempts to use environment variable key names not starting with `NEXT_PUBLIC_`. However, it was not possible to get the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret` for authentication during OAuth authentication, and even `SITE_TITLE` and `BASE_DIRECTORY` were not set as key values of environment variables. For the sake of smooth deployment, these parameters had to temporarily use environment variable key names starting with `NEXT_PUBLIC_`. ## Todo List @@ -76,7 +82,9 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - Deepen the study of the original version of the code and strive to implement the function with environment variable key names that do not start with `NEXT_PUBLIC_`, to improve security. -- Close the OAuth authentication pathway after completing the OAuth authentication, or optimize the OAuth authentication process, and strive not to leak the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret`. +- Close the OAuth authentication pathway after completing the OAuth authentication, striving not to leak the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret`. + +> It has been preliminarily achieved, but further verification is needed to determine whether the related security risks have been removed. - Redesign the LOGO. The contrast of the original LOGO is too low, and it is not consistent enough with the style of other icons and fonts on the page. From e33b44b955612ce506856dbeafe852ec215724c3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 1 Jul 2023 01:37:42 +0800 Subject: [PATCH 25/88] Update README.zh-CN.md --- README.zh-CN.md | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index ca36cc1771..3951504937 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -74,7 +74,7 @@ > 本版本使用的环境变量均以`NEXT_PUBLIC_`开头(否则无法正常运行)…… > -> 在最开始有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,无法获取到`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值而不能通过认证,包括`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 +> 在最开始有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,无法获取到`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值而不能通过认证,甚至`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 ## 待办备忘 @@ -84,7 +84,9 @@ - 深入研究原版本代码,争取不以`NEXT_PUBLIC_`开头的环境变量键名实现功能,以提高安全性。 -- 在完成OAuth认证后,关闭OAuth认证通道(目前已初步完成,但相关安全风险有无解除还有待进一步检验),争取不泄露`USER_PRINCIPLE_NAME`~~、`clientId`和`obfuscatedClientSecret`~~的值。 +- 在完成OAuth认证后,关闭OAuth认证通,争取不泄露`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值。 + +> 目前已初步完成,但相关安全风险有无解除还有待进一步检验。 - 重新设计LOGO,原LOGO对比度太低,与页面中其他图标和字体风格不够一致。 From bac902a79903a5d4e6d36c3561bf5b37f0960a1f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Mon, 3 Jul 2023 18:13:33 +0800 Subject: [PATCH 26/88] =?UTF-8?q?=E6=9B=B4=E6=96=B0=20step-3.tsx?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../onedrive-vercel-index-oauth/step-3.tsx | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index b21af3a463..055fa6690f 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -12,6 +12,7 @@ import Footer from '../../components/Footer' import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer } from '../../utils/oAuthHandler' import { LoadingIcon } from '../../components/Loading' +import { getAccessToken } from '../api' // 导入getAccessToken函数 export default function OAuthStep3({ accessToken, expiryTime, refreshToken, error, description, errorUri }) { const router = useRouter() @@ -226,7 +227,19 @@ export default function OAuthStep3({ accessToken, expiryTime, refreshToken, erro export async function getServerSideProps({ query, locale }) { const { authCode } = query - // Return if no auth code is present + // 检查是否已经通过OAuth认证 + const accessToken = await getAccessToken(); + if (accessToken) { + // 如果已经通过OAuth认证,重定向到主页 + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + + // 如果没有通过OAuth认证,继续执行OAuth认证的流程 if (!authCode) { return { props: { @@ -251,7 +264,7 @@ export async function getServerSideProps({ query, locale }) { } } - const { expiryTime, accessToken, refreshToken } = response + const { expiryTime, refreshToken } = response return { props: { From f46e6db94565591f21fd575c5ff07638b4d6a4fc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Tue, 4 Jul 2023 14:23:00 +0800 Subject: [PATCH 27/88] Update site.config.js --- config/site.config.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/config/site.config.js b/config/site.config.js index b66cac4f6d..e8184b8d2d 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -38,11 +38,11 @@ module.exports = { // [OPTIONAL] The footer component of your website. You can write HTML here, but you need to escape double // quotes - changing " to \". You can write anything here, and if you like badges, generate some with https://shields.io footer: - 'Powered by onedrive-vercel-index | Made by SpencerWoo | Modified by iRedScarf | MIT License', + 'Powered by onedrive-vercel-index.', // [OPTIONAL] This is where you specify the folders that are password protected. It is an array of paths pointing to all // the directories in which you have .password set. Check the documentation for details. - protectedRoutes: [], + protectedRoutes: process.env.NEXT_PUBLIC_PROTECTED_ROUTES ? process.env.NEXT_PUBLIC_PROTECTED_ROUTES.split(',') : [], // [OPTIONAL] Use "" here if you want to remove this email address from the nav bar. email: '', From 93592514b21be01e82672468910a5ea5eb26509f Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Wed, 5 Jul 2023 15:06:07 +0800 Subject: [PATCH 28/88] Modifications Transition some settings to environment variables. --- README.zh-CN.md | 36 +----- config/api.config.js | 10 -- config/site.config.js | 4 +- src/pages/api/config.ts | 10 ++ src/pages/api/index.ts | 6 +- .../onedrive-vercel-index-oauth/step-1.tsx | 50 ++++---- .../onedrive-vercel-index-oauth/step-3.tsx | 113 +++++++++--------- src/utils/oAuthHandler.ts | 8 +- src/utils/protectedRouteHandler.ts | 2 +- 9 files changed, 111 insertions(+), 128 deletions(-) create mode 100644 src/pages/api/config.ts mode change 100644 => 100755 src/pages/api/index.ts mode change 100755 => 100644 src/pages/onedrive-vercel-index-oauth/step-1.tsx diff --git a/README.zh-CN.md b/README.zh-CN.md index 3951504937..57d299bc6c 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,7 +36,7 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,REDIS_URL) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 @@ -54,41 +54,15 @@ - 本版本对比原版主要是把需要先修改`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`,以及修改`config/site.config.js`中的`userPrincipalName`、`title`和`baseDirectory`的步骤提取出来,放在Vercel部署时的环境变量设置中进行。 -- 在进行OAuth认证第一步的页面中隐藏了`clientId`和`obfuscatedClientSecret`的具体值,只显示值的前6位和后6位用于核对。 - -- 在OAuth认证的第一步和第二步页面代码中,添加了检查是否已通过OAuth认证的逻辑,若网站已完成OAuth认证,则重新定向到首页,否则正常进行认证流程。 - -- 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改,本版本的Demo显示了Email图标),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 +- 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 - 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 -## 安全风险 - -- 在本版本和原作者的归档版本中,都会在网页的源代码中泄露部署者的OneDrive帐号`USER_PRINCIPLE_NAME`。 - -- 原作者的归档版本中,可在OAuth认证第二步用来获取授权码的链接中查看到部署者的`clientId`,`obfuscatedClientSecret`则可在OAuth认证第一步的源代码中查看到。 - -> 本版本初步尝试在进行OAuth认证流程时检查是否已通过认证,若已通过则重定向至首页,否则才进行OAuth认证流程,设想如此来阻止有心之人通过OAuth认证的链接地址来获取`clientId`和`obfuscatedClientSecret`的值。 - -- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 - -> 本版本使用的环境变量均以`NEXT_PUBLIC_`开头(否则无法正常运行)…… -> -> 在最开始有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,无法获取到`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值而不能通过认证,甚至`SITE_TITLE`和`BASE_DIRECTORY`也都不是环境变量的键值设置。为了顺利部署,只好暂时把这些参数都使用以`NEXT_PUBLIC_`开头的环境变量键名了。 - -## 待办备忘 - -- 把`config/site.config.js`中的加密文件夹`protectedRoutes`参数放在环境变量中设置。 - -- 把密码放在环境变量而非`.password`文件。 - -- 深入研究原版本代码,争取不以`NEXT_PUBLIC_`开头的环境变量键名实现功能,以提高安全性。 - -- 在完成OAuth认证后,关闭OAuth认证通,争取不泄露`USER_PRINCIPLE_NAME`、`clientId`和`obfuscatedClientSecret`的值。 +## 安全风险和一些问题 -> 目前已初步完成,但相关安全风险有无解除还有待进一步检验。 +- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。应该避免在以`NEXT_PUBLIC_`开头的环境变量中存储敏感信息,如API密钥或数据库密码。这些信息应该只在服务器端代码中使用,并且应该使用不带`NEXT_PUBLIC_`前缀的环境变量来存储。 -- 重新设计LOGO,原LOGO对比度太低,与页面中其他图标和字体风格不够一致。 +- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`obfuscatedClientSecret`在没有OneDrive帐户的登录密码时,也不算是太敏感的信息,就暂时这样解决了。 ## License diff --git a/config/api.config.js b/config/api.config.js index ccc50bcac6..0d33cd315d 100644 --- a/config/api.config.js +++ b/config/api.config.js @@ -7,20 +7,10 @@ * - If you are using a E5 Subscription OneDrive for Business account, the direct links of your files are not the same here. * In which case you would need to change directLinkRegex. */ -const clientIdEnv = process.env.NEXT_PUBLIC_CLIENT_ID; -if (!clientIdEnv) { - throw new Error('CLIENT_ID is not defined in api.config.js'); -} -const clientSecretEnv = process.env.NEXT_PUBLIC_CLIENT_SECRET; -if (!clientSecretEnv) { - throw new Error('CLIENT_SECRET is not defined in api.config.js'); -} module.exports = { // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. - clientId: clientIdEnv, - obfuscatedClientSecret: clientSecretEnv, // The redirectUri is the URL that the user will be redirected to after they have authenticated with Microsoft Graph API. // Likewise, you would not need to change redirectUri if you are using your personal Microsoft account with OneDrive International. diff --git a/config/site.config.js b/config/site.config.js index e8184b8d2d..bd4187bf55 100644 --- a/config/site.config.js +++ b/config/site.config.js @@ -7,7 +7,6 @@ module.exports = { // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about // your email being exposed in public. - userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPLE_NAME, // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. @@ -20,7 +19,6 @@ module.exports = { title: process.env.NEXT_PUBLIC_SITE_TITLE || 'OneDrive-Vercel-Index', // The folder that you are to share publicly with onedrive-vercel-index. Use '/' if you want to share your root folder. - baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. @@ -45,7 +43,7 @@ module.exports = { protectedRoutes: process.env.NEXT_PUBLIC_PROTECTED_ROUTES ? process.env.NEXT_PUBLIC_PROTECTED_ROUTES.split(',') : [], // [OPTIONAL] Use "" here if you want to remove this email address from the nav bar. - email: '', + email: 'mailto:iredscarf@freeloop.one', // [OPTIONAL] This is an array of names and links for setting your social information and links. // In the latest update, all brand icons inside font awesome is supported and the icon to render is based on the name diff --git a/src/pages/api/config.ts b/src/pages/api/config.ts new file mode 100644 index 0000000000..b50817140d --- /dev/null +++ b/src/pages/api/config.ts @@ -0,0 +1,10 @@ +import type { NextApiRequest, NextApiResponse } from 'next' + +export default function handler(req: NextApiRequest, res: NextApiResponse) { + res.status(200).json({ + clientId: process.env.CLIENT_ID || '', + clientSecret: process.env.CLIENT_SECRET || '', + userPrincipalName: process.env.USER_PRINCIPLE_NAME || '', + baseDirectory: process.env.BASE_DIRECTORY || '/' + }) +} diff --git a/src/pages/api/index.ts b/src/pages/api/index.ts old mode 100644 new mode 100755 index ed80e7ac8b..017fd0825b --- a/src/pages/api/index.ts +++ b/src/pages/api/index.ts @@ -10,8 +10,8 @@ import { compareHashedToken } from '../../utils/protectedRouteHandler' import { getOdAuthTokens, storeOdAuthTokens } from '../../utils/odAuthTokenStore' import { runCorsMiddleware } from './raw' -const basePath = pathPosix.resolve('/', siteConfig.baseDirectory) -const clientSecret = revealObfuscatedToken(apiConfig.obfuscatedClientSecret) +const basePath = pathPosix.resolve('/', process.env.BASE_DIRECTORY || '/') +const clientSecret = revealObfuscatedToken(process.env.CLIENT_SECRET || '') /** * Encode the path of the file relative to the base directory @@ -50,7 +50,7 @@ export async function getAccessToken(): Promise { // Fetch new access token with in storage refresh token const body = new URLSearchParams() - body.append('client_id', apiConfig.clientId) + body.append('client_id', process.env.CLIENT_ID || '') body.append('redirect_uri', apiConfig.redirectUri) body.append('client_secret', clientSecret) body.append('refresh_token', refreshToken) diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx old mode 100755 new mode 100644 index 3eb78cdd86..1380083fe9 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -20,7 +20,32 @@ function obfuscateSensitiveData(data) { return `${start}******${end}`; } -export default function OAuthStep1() { +export async function getServerSideProps({ locale }) { + const clientId = process.env.CLIENT_ID || ''; + const clientSecret = process.env.CLIENT_SECRET || ''; + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + + // 如果访问令牌不存在,正常渲染页面 + return { + props: { + ...(await serverSideTranslations(locale, ['common'])), + clientId, + clientSecret, + }, + } +} + +export default function OAuthStep1({ clientId, clientSecret }) { const router = useRouter() const { t } = useTranslation() @@ -83,7 +108,7 @@ export default function OAuthStep1() { CLIENT_ID - {obfuscateSensitiveData(apiConfig.clientId)} + {obfuscateSensitiveData(clientId)} @@ -91,7 +116,7 @@ export default function OAuthStep1() { CLIENT_SECRET* - {obfuscateSensitiveData(apiConfig.obfuscatedClientSecret)} + {obfuscateSensitiveData(clientSecret)} @@ -156,22 +181,3 @@ export default function OAuthStep1() { ) } - -export async function getServerSideProps({ locale }) { - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - // 如果访问令牌存在,重定向到主页 - if (accessToken) { - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - // 如果访问令牌不存在,正常渲染页面 - return { - props: { - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 055fa6690f..9a4b9a6d7d 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -14,7 +14,63 @@ import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer } from ' import { LoadingIcon } from '../../components/Loading' import { getAccessToken } from '../api' // 导入getAccessToken函数 -export default function OAuthStep3({ accessToken, expiryTime, refreshToken, error, description, errorUri }) { +export async function getServerSideProps({ query, locale }) { + const { authCode } = query + + // 检查是否已经通过OAuth认证 + const accessToken = await getAccessToken(); + if (accessToken) { + // 如果已经通过OAuth认证,重定向到主页 + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + + // 如果没有通过OAuth认证,继续执行OAuth认证的流程 + if (!authCode) { + return { + props: { + error: 'No auth code present', + description: 'Where is the auth code? Did you follow step 2 you silly donut?', + userPrincipalName: process.env.USER_PRINCIPLE_NAME || '', + ...(await serverSideTranslations(locale, ['common'])), + }, + } + } + + const response = await requestTokenWithAuthCode(authCode) + + // If error response, return invalid + if ('error' in response) { + return { + props: { + error: response.error, + description: response.errorDescription, + errorUri: response.errorUri, + userPrincipalName: process.env.USER_PRINCIPLE_NAME || '', + ...(await serverSideTranslations(locale, ['common'])), + }, + } + } + + const { expiryTime, refreshToken } = response + + return { + props: { + error: null, + expiryTime, + accessToken, + refreshToken, + userPrincipalName: process.env.USER_PRINCIPLE_NAME || '', + ...(await serverSideTranslations(locale, ['common'])), + }, + } +} + +export default function OAuthStep3({ userPrincipalName, accessToken, expiryTime, refreshToken, error, description, errorUri }) { const router = useRouter() const [expiryTimeLeft, setExpiryTimeLeft] = useState(expiryTime) @@ -56,7 +112,7 @@ export default function OAuthStep3({ accessToken, expiryTime, refreshToken, erro ) return } - if (data.userPrincipalName !== siteConfig.userPrincipalName) { + if (data.userPrincipalName !== userPrincipalName) { setButtonError(true) setButtonContent(
@@ -223,56 +279,3 @@ export default function OAuthStep3({ accessToken, expiryTime, refreshToken, erro
) } - -export async function getServerSideProps({ query, locale }) { - const { authCode } = query - - // 检查是否已经通过OAuth认证 - const accessToken = await getAccessToken(); - if (accessToken) { - // 如果已经通过OAuth认证,重定向到主页 - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - - // 如果没有通过OAuth认证,继续执行OAuth认证的流程 - if (!authCode) { - return { - props: { - error: 'No auth code present', - description: 'Where is the auth code? Did you follow step 2 you silly donut?', - ...(await serverSideTranslations(locale, ['common'])), - }, - } - } - - const response = await requestTokenWithAuthCode(authCode) - - // If error response, return invalid - if ('error' in response) { - return { - props: { - error: response.error, - description: response.errorDescription, - errorUri: response.errorUri, - ...(await serverSideTranslations(locale, ['common'])), - }, - } - } - - const { expiryTime, refreshToken } = response - - return { - props: { - error: null, - expiryTime, - accessToken, - refreshToken, - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 4fde3d5d3d..f2e838fbe3 100644 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -19,7 +19,8 @@ export function revealObfuscatedToken(obfuscated: string): string { // Generate the Microsoft OAuth 2.0 authorization URL, used for requesting the authorisation code export function generateAuthorisationUrl(): string { - const { clientId, redirectUri, authApi, scope } = apiConfig + const clientId = process.env.CLIENT_ID || ''; + const { redirectUri, authApi, scope } = apiConfig const authUrl = authApi.replace('/token', '/authorize') // Construct URL parameters for OAuth2 @@ -55,8 +56,9 @@ export async function requestTokenWithAuthCode( | { expiryTime: string; accessToken: string; refreshToken: string } | { error: string; errorDescription: string; errorUri: string } > { - const { clientId, redirectUri, authApi } = apiConfig - const clientSecret = revealObfuscatedToken(apiConfig.obfuscatedClientSecret) + const clientId = process.env.CLIENT_ID || '' + const clientSecret = revealObfuscatedToken(process.env.CLIENT_SECRET || '') + const { redirectUri, authApi } = apiConfig // Construct URL parameters for OAuth2 const params = new URLSearchParams() diff --git a/src/utils/protectedRouteHandler.ts b/src/utils/protectedRouteHandler.ts index 6e2bbdb333..afa99b625a 100644 --- a/src/utils/protectedRouteHandler.ts +++ b/src/utils/protectedRouteHandler.ts @@ -55,4 +55,4 @@ export function matchProtectedRoute(route: string): string { } } return authTokenPath -} +} \ No newline at end of file From b4cf4b42d63ab97f60c956a82ca19ada56250cdf Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Wed, 5 Jul 2023 15:09:30 +0800 Subject: [PATCH 29/88] Update README.zh-CN.md --- README.zh-CN.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 57d299bc6c..86f048bb45 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,7 +36,7 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,REDIS_URL) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,REDIS_URL) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 From 6fd42e4cb2991a399a6feb7b80b9af64c2f41042 Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Wed, 5 Jul 2023 16:21:15 +0800 Subject: [PATCH 30/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index f2e838fbe3..aeb594b0dc 100644 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -3,6 +3,11 @@ import CryptoJS from 'crypto-js' import apiConfig from '../../config/api.config' +async function getConfig() { + const res = await fetch('/api/config') + return await res.json() +} + // Just a disguise to obfuscate required tokens (including but not limited to client secret, // access tokens, and refresh tokens), used along with the following two functions const AES_SECRET_KEY = 'onedrive-vercel-index' @@ -19,7 +24,8 @@ export function revealObfuscatedToken(obfuscated: string): string { // Generate the Microsoft OAuth 2.0 authorization URL, used for requesting the authorisation code export function generateAuthorisationUrl(): string { - const clientId = process.env.CLIENT_ID || ''; + const config = await getConfig() + const clientId = config.clientId const { redirectUri, authApi, scope } = apiConfig const authUrl = authApi.replace('/token', '/authorize') @@ -56,8 +62,9 @@ export async function requestTokenWithAuthCode( | { expiryTime: string; accessToken: string; refreshToken: string } | { error: string; errorDescription: string; errorUri: string } > { - const clientId = process.env.CLIENT_ID || '' - const clientSecret = revealObfuscatedToken(process.env.CLIENT_SECRET || '') + const config = await getConfig() + const clientId = config.clientId + const clientSecret = revealObfuscatedToken(config.clientSecret) const { redirectUri, authApi } = apiConfig // Construct URL parameters for OAuth2 From 25520889d0fcb0cbdb781fb36bfa469b0f0505cd Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Thu, 6 Jul 2023 10:24:09 +0800 Subject: [PATCH 31/88] update --- README.zh-CN.md | 2 +- src/pages/onedrive-vercel-index-oauth/step-2.tsx | 14 +++++++++++--- src/utils/oAuthHandler.ts | 7 ++++--- 3 files changed, 16 insertions(+), 7 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 86f048bb45..f31d1df013 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,7 +36,7 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,REDIS_URL) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 diff --git a/src/pages/onedrive-vercel-index-oauth/step-2.tsx b/src/pages/onedrive-vercel-index-oauth/step-2.tsx index dd4fd70e00..70ebd2c5e2 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-2.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-2.tsx @@ -1,7 +1,7 @@ import Head from 'next/head' import Image from 'next/image' import { useRouter } from 'next/router' -import { useState } from 'react' +import { useState, useEffect } from 'react' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' import { useTranslation, Trans } from 'next-i18next' import { serverSideTranslations } from 'next-i18next/serverSideTranslations' @@ -22,7 +22,13 @@ export default function OAuthStep2() { const { t } = useTranslation() - const oAuthUrl = generateAuthorisationUrl() + // const oAuthUrl = generateAuthorisationUrl() + + const [oAuthUrl, setOAuthUrl] = useState(null) + + useEffect(() => { + generateAuthorisationUrl().then(url => setOAuthUrl(url)) + }, []) return (
@@ -60,7 +66,9 @@ export default function OAuthStep2() {
{ - window.open(oAuthUrl) + if (oAuthUrl) { + window.open(oAuthUrl) + } }} >
diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index aeb594b0dc..ac83d96984 100644 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -4,8 +4,9 @@ import CryptoJS from 'crypto-js' import apiConfig from '../../config/api.config' async function getConfig() { - const res = await fetch('/api/config') - return await res.json() + const res = await axios.get('/api/config') + const data = typeof res.data === 'string' ? JSON.parse(res.data) : res.data + return data } // Just a disguise to obfuscate required tokens (including but not limited to client secret, @@ -23,7 +24,7 @@ export function revealObfuscatedToken(obfuscated: string): string { } // Generate the Microsoft OAuth 2.0 authorization URL, used for requesting the authorisation code -export function generateAuthorisationUrl(): string { +export async function generateAuthorisationUrl(): Promise { const config = await getConfig() const clientId = config.clientId const { redirectUri, authApi, scope } = apiConfig From d819357d821cf9137721de0711c6b6b279e09355 Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Fri, 7 Jul 2023 11:07:54 +0800 Subject: [PATCH 32/88] update --- .../onedrive-vercel-index-oauth/step-1.tsx | 4 +- .../onedrive-vercel-index-oauth/step-2.tsx | 38 +++++++++---------- .../onedrive-vercel-index-oauth/step-3.tsx | 26 ++++++------- src/utils/oAuthHandler.ts | 3 +- 4 files changed, 34 insertions(+), 37 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx index 1380083fe9..783b28fe26 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -24,7 +24,7 @@ export async function getServerSideProps({ locale }) { const clientId = process.env.CLIENT_ID || ''; const clientSecret = process.env.CLIENT_SECRET || ''; const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - + // 如果访问令牌存在,重定向到主页 if (accessToken) { return { @@ -34,7 +34,7 @@ export async function getServerSideProps({ locale }) { }, } } - + // 如果访问令牌不存在,正常渲染页面 return { props: { diff --git a/src/pages/onedrive-vercel-index-oauth/step-2.tsx b/src/pages/onedrive-vercel-index-oauth/step-2.tsx index 70ebd2c5e2..6782223e8c 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-2.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-2.tsx @@ -13,6 +13,25 @@ import { LoadingIcon } from '../../components/Loading' import { extractAuthCodeFromRedirected, generateAuthorisationUrl } from '../../utils/oAuthHandler' import { getAccessToken } from '../api' // 导入getAccessToken函数 +export async function getServerSideProps({ locale }) { + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + // 如果访问令牌不存在,正常渲染页面 + return { + props: { + ...(await serverSideTranslations(locale, ['common'])), + }, + } +} + export default function OAuthStep2() { const router = useRouter() @@ -149,22 +168,3 @@ export default function OAuthStep2() {
) } - -export async function getServerSideProps({ locale }) { - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - // 如果访问令牌存在,重定向到主页 - if (accessToken) { - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - // 如果访问令牌不存在,正常渲染页面 - return { - props: { - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 9a4b9a6d7d..9c7655d4c5 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -16,18 +16,19 @@ import { getAccessToken } from '../api' // 导入getAccessToken函数 export async function getServerSideProps({ query, locale }) { const { authCode } = query + const userPrincipalName = process.env.USER_PRINCIPLE_NAME || ''; // 检查是否已经通过OAuth认证 - const accessToken = await getAccessToken(); - if (accessToken) { - // 如果已经通过OAuth认证,重定向到主页 - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } + //const accessToken = await getAccessToken(); + //if (accessToken) { + //如果已经通过OAuth认证,重定向到主页 + //return { + //redirect: { + //destination: '/', + //permanent: false, + //}, + //} + //} // 如果没有通过OAuth认证,继续执行OAuth认证的流程 if (!authCode) { @@ -35,7 +36,6 @@ export async function getServerSideProps({ query, locale }) { props: { error: 'No auth code present', description: 'Where is the auth code? Did you follow step 2 you silly donut?', - userPrincipalName: process.env.USER_PRINCIPLE_NAME || '', ...(await serverSideTranslations(locale, ['common'])), }, } @@ -50,13 +50,12 @@ export async function getServerSideProps({ query, locale }) { error: response.error, description: response.errorDescription, errorUri: response.errorUri, - userPrincipalName: process.env.USER_PRINCIPLE_NAME || '', ...(await serverSideTranslations(locale, ['common'])), }, } } - const { expiryTime, refreshToken } = response + const { expiryTime, accessToken, refreshToken } = response return { props: { @@ -64,7 +63,6 @@ export async function getServerSideProps({ query, locale }) { expiryTime, accessToken, refreshToken, - userPrincipalName: process.env.USER_PRINCIPLE_NAME || '', ...(await serverSideTranslations(locale, ['common'])), }, } diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index ac83d96984..8197141db5 100644 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -5,8 +5,7 @@ import apiConfig from '../../config/api.config' async function getConfig() { const res = await axios.get('/api/config') - const data = typeof res.data === 'string' ? JSON.parse(res.data) : res.data - return data + return res.data } // Just a disguise to obfuscate required tokens (including but not limited to client secret, From 839eae934f7bd4c1c6e1c31ffe6c5481ad4af36a Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Fri, 7 Jul 2023 12:31:44 +0800 Subject: [PATCH 33/88] Update step-3.tsx --- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 9c7655d4c5..16e723f426 100644 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -59,6 +59,7 @@ export async function getServerSideProps({ query, locale }) { return { props: { + userPrincipalName, error: null, expiryTime, accessToken, From 083fdbb981e4be71c6e6953ad50f08cf69ef6150 Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Fri, 7 Jul 2023 15:39:08 +0800 Subject: [PATCH 34/88] rollback --- config/api.config.js | 11 ++ config/site.config.js | 6 +- i18next-parser.config.js | 0 i18next.d.ts | 0 next-env.d.ts | 0 next-i18next.config.js | 0 next.config.js | 0 package.json | 0 pnpm-lock.yaml | 0 postcss.config.js | 0 public/demo.png | Bin public/footer.png | Bin 0 -> 3549 bytes public/header.png | Bin 0 -> 72302 bytes public/icons/128.png | Bin public/icons/256.png | Bin public/icons/512.png | Bin public/icons/64.png | Bin public/images/fabulous-celebration.png | Bin public/images/fabulous-come-back-later.png | Bin public/images/fabulous-fireworks.png | Bin public/images/fabulous-page-not-found.png | Bin public/images/fabulous-rip-2.png | Bin public/images/fabulous-wapmire-weekdays.png | Bin public/images/step-2-screenshot.png | Bin public/locales/de-DE/common.json | 0 public/locales/en/common.json | 0 public/locales/es/common.json | 0 public/locales/hi/common.json | 0 public/locales/id/common.json | 0 public/locales/tr-TR/common.json | 0 public/locales/zh-CN/common.json | 0 public/locales/zh-TW/common.json | 0 public/players/iina.png | Bin public/players/nplayer.png | Bin public/players/potplayer.png | Bin public/players/vlc.png | Bin renovate.json | 0 src/components/Auth.tsx | 0 src/components/Breadcrumb.tsx | 0 src/components/CustomEmbedLinkMenu.tsx | 0 src/components/DownloadBtnGtoup.tsx | 0 src/components/FileListing.tsx | 0 src/components/FolderGridLayout.tsx | 0 src/components/FolderListLayout.tsx | 0 src/components/Footer.tsx | 0 src/components/FourOhFour.tsx | 0 src/components/Loading.tsx | 0 src/components/MultiFileDownloader.tsx | 0 src/components/SearchModal.tsx | 0 src/components/SwitchLang.tsx | 0 src/components/SwitchLayout.tsx | 0 src/components/previews/AudioPreview.tsx | 0 src/components/previews/CodePreview.tsx | 0 src/components/previews/Containers.tsx | 0 src/components/previews/DefaultPreview.tsx | 0 src/components/previews/EPUBPreview.tsx | 0 src/components/previews/ImagePreview.tsx | 0 src/components/previews/MarkdownPreview.tsx | 0 src/components/previews/OfficePreview.tsx | 0 src/components/previews/PDFPreview.tsx | 0 src/components/previews/TextPreview.tsx | 0 src/components/previews/URLPreview.tsx | 0 src/components/previews/VideoPreview.tsx | 0 src/pages/[...path].tsx | 0 src/pages/_app.tsx | 0 src/pages/_document.tsx | 0 src/pages/api/config.ts | 10 -- src/pages/api/index.ts | 6 +- src/pages/api/item.ts | 0 src/pages/api/name/[name].ts | 0 src/pages/api/raw.ts | 0 src/pages/api/search.ts | 0 src/pages/api/thumbnail.ts | 0 src/pages/index.tsx | 0 .../onedrive-vercel-index-oauth/step-1.tsx | 60 ++++------ .../onedrive-vercel-index-oauth/step-2.tsx | 52 ++++---- .../onedrive-vercel-index-oauth/step-3.tsx | 113 +++++++++--------- src/styles/globals.css | 0 src/styles/markdown-github.css | 0 src/types/index.d.ts | 0 src/utils/fetchOnMount.ts | 0 src/utils/fetchWithSWR.ts | 0 src/utils/fileDetails.ts | 0 src/utils/getBaseUrl.ts | 0 src/utils/getFileIcon.ts | 0 src/utils/getPreviewType.ts | 0 src/utils/getReadablePath.ts | 0 src/utils/oAuthHandler.ts | 17 +-- src/utils/odAuthTokenStore.ts | 0 src/utils/protectedRouteHandler.ts | 2 +- src/utils/useDeviceOS.ts | 0 src/utils/useLocalStorage.ts | 0 tailwind.config.js | 0 tsconfig.json | 0 94 files changed, 124 insertions(+), 153 deletions(-) mode change 100644 => 100755 config/api.config.js mode change 100644 => 100755 config/site.config.js mode change 100644 => 100755 i18next-parser.config.js mode change 100644 => 100755 i18next.d.ts mode change 100644 => 100755 next-env.d.ts mode change 100644 => 100755 next-i18next.config.js mode change 100644 => 100755 next.config.js mode change 100644 => 100755 package.json mode change 100644 => 100755 pnpm-lock.yaml mode change 100644 => 100755 postcss.config.js mode change 100644 => 100755 public/demo.png create mode 100755 public/footer.png create mode 100755 public/header.png mode change 100644 => 100755 public/icons/128.png mode change 100644 => 100755 public/icons/256.png mode change 100644 => 100755 public/icons/512.png mode change 100644 => 100755 public/icons/64.png mode change 100644 => 100755 public/images/fabulous-celebration.png mode change 100644 => 100755 public/images/fabulous-come-back-later.png mode change 100644 => 100755 public/images/fabulous-fireworks.png mode change 100644 => 100755 public/images/fabulous-page-not-found.png mode change 100644 => 100755 public/images/fabulous-rip-2.png mode change 100644 => 100755 public/images/fabulous-wapmire-weekdays.png mode change 100644 => 100755 public/images/step-2-screenshot.png mode change 100644 => 100755 public/locales/de-DE/common.json mode change 100644 => 100755 public/locales/en/common.json mode change 100644 => 100755 public/locales/es/common.json mode change 100644 => 100755 public/locales/hi/common.json mode change 100644 => 100755 public/locales/id/common.json mode change 100644 => 100755 public/locales/tr-TR/common.json mode change 100644 => 100755 public/locales/zh-CN/common.json mode change 100644 => 100755 public/locales/zh-TW/common.json mode change 100644 => 100755 public/players/iina.png mode change 100644 => 100755 public/players/nplayer.png mode change 100644 => 100755 public/players/potplayer.png mode change 100644 => 100755 public/players/vlc.png mode change 100644 => 100755 renovate.json mode change 100644 => 100755 src/components/Auth.tsx mode change 100644 => 100755 src/components/Breadcrumb.tsx mode change 100644 => 100755 src/components/CustomEmbedLinkMenu.tsx mode change 100644 => 100755 src/components/DownloadBtnGtoup.tsx mode change 100644 => 100755 src/components/FileListing.tsx mode change 100644 => 100755 src/components/FolderGridLayout.tsx mode change 100644 => 100755 src/components/FolderListLayout.tsx mode change 100644 => 100755 src/components/Footer.tsx mode change 100644 => 100755 src/components/FourOhFour.tsx mode change 100644 => 100755 src/components/Loading.tsx mode change 100644 => 100755 src/components/MultiFileDownloader.tsx mode change 100644 => 100755 src/components/SearchModal.tsx mode change 100644 => 100755 src/components/SwitchLang.tsx mode change 100644 => 100755 src/components/SwitchLayout.tsx mode change 100644 => 100755 src/components/previews/AudioPreview.tsx mode change 100644 => 100755 src/components/previews/CodePreview.tsx mode change 100644 => 100755 src/components/previews/Containers.tsx mode change 100644 => 100755 src/components/previews/DefaultPreview.tsx mode change 100644 => 100755 src/components/previews/EPUBPreview.tsx mode change 100644 => 100755 src/components/previews/ImagePreview.tsx mode change 100644 => 100755 src/components/previews/MarkdownPreview.tsx mode change 100644 => 100755 src/components/previews/OfficePreview.tsx mode change 100644 => 100755 src/components/previews/PDFPreview.tsx mode change 100644 => 100755 src/components/previews/TextPreview.tsx mode change 100644 => 100755 src/components/previews/URLPreview.tsx mode change 100644 => 100755 src/components/previews/VideoPreview.tsx mode change 100644 => 100755 src/pages/[...path].tsx mode change 100644 => 100755 src/pages/_app.tsx mode change 100644 => 100755 src/pages/_document.tsx delete mode 100644 src/pages/api/config.ts mode change 100644 => 100755 src/pages/api/item.ts mode change 100644 => 100755 src/pages/api/name/[name].ts mode change 100644 => 100755 src/pages/api/raw.ts mode change 100644 => 100755 src/pages/api/search.ts mode change 100644 => 100755 src/pages/api/thumbnail.ts mode change 100644 => 100755 src/pages/index.tsx mode change 100644 => 100755 src/pages/onedrive-vercel-index-oauth/step-1.tsx mode change 100644 => 100755 src/pages/onedrive-vercel-index-oauth/step-2.tsx mode change 100644 => 100755 src/pages/onedrive-vercel-index-oauth/step-3.tsx mode change 100644 => 100755 src/styles/globals.css mode change 100644 => 100755 src/styles/markdown-github.css mode change 100644 => 100755 src/types/index.d.ts mode change 100644 => 100755 src/utils/fetchOnMount.ts mode change 100644 => 100755 src/utils/fetchWithSWR.ts mode change 100644 => 100755 src/utils/fileDetails.ts mode change 100644 => 100755 src/utils/getBaseUrl.ts mode change 100644 => 100755 src/utils/getFileIcon.ts mode change 100644 => 100755 src/utils/getPreviewType.ts mode change 100644 => 100755 src/utils/getReadablePath.ts mode change 100644 => 100755 src/utils/oAuthHandler.ts mode change 100644 => 100755 src/utils/odAuthTokenStore.ts mode change 100644 => 100755 src/utils/protectedRouteHandler.ts mode change 100644 => 100755 src/utils/useDeviceOS.ts mode change 100644 => 100755 src/utils/useLocalStorage.ts mode change 100644 => 100755 tailwind.config.js mode change 100644 => 100755 tsconfig.json diff --git a/config/api.config.js b/config/api.config.js old mode 100644 new mode 100755 index 0d33cd315d..2e2d047d75 --- a/config/api.config.js +++ b/config/api.config.js @@ -8,9 +8,20 @@ * In which case you would need to change directLinkRegex. */ +const clientIdEnv = process.env.NEXT_PUBLIC_CLIENT_ID; +if (!clientIdEnv) { + throw new Error('`clientId` is not defined in api.config.js'); +} +const clientSecretEnv = process.env.NEXT_PUBLIC_CLIENT_SECRET; +if (!clientSecretEnv) { + throw new Error('`clientSecret` is not defined in api.config.js'); +} + module.exports = { // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. + clientId: clientIdEnv, + obfuscatedClientSecret: clientSecretEnv, // The redirectUri is the URL that the user will be redirected to after they have authenticated with Microsoft Graph API. // Likewise, you would not need to change redirectUri if you are using your personal Microsoft account with OneDrive International. diff --git a/config/site.config.js b/config/site.config.js old mode 100644 new mode 100755 index bd4187bf55..6236c04843 --- a/config/site.config.js +++ b/config/site.config.js @@ -7,6 +7,7 @@ module.exports = { // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about // your email being exposed in public. + userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPLE_NAME || '', // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. @@ -19,10 +20,11 @@ module.exports = { title: process.env.NEXT_PUBLIC_SITE_TITLE || 'OneDrive-Vercel-Index', // The folder that you are to share publicly with onedrive-vercel-index. Use '/' if you want to share your root folder. + baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. - maxItems: 200, + maxItems: 100, // [OPTIONAL] We use Google Fonts natively for font customisations. // You can check and generate the required links and names at https://fonts.google.com. @@ -36,7 +38,7 @@ module.exports = { // [OPTIONAL] The footer component of your website. You can write HTML here, but you need to escape double // quotes - changing " to \". You can write anything here, and if you like badges, generate some with https://shields.io footer: - 'Powered by onedrive-vercel-index.', + 'Powered by onedrive-vercel-index.', // [OPTIONAL] This is where you specify the folders that are password protected. It is an array of paths pointing to all // the directories in which you have .password set. Check the documentation for details. diff --git a/i18next-parser.config.js b/i18next-parser.config.js old mode 100644 new mode 100755 diff --git a/i18next.d.ts b/i18next.d.ts old mode 100644 new mode 100755 diff --git a/next-env.d.ts b/next-env.d.ts old mode 100644 new mode 100755 diff --git a/next-i18next.config.js b/next-i18next.config.js old mode 100644 new mode 100755 diff --git a/next.config.js b/next.config.js old mode 100644 new mode 100755 diff --git a/package.json b/package.json old mode 100644 new mode 100755 diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml old mode 100644 new mode 100755 diff --git a/postcss.config.js b/postcss.config.js old mode 100644 new mode 100755 diff --git a/public/demo.png b/public/demo.png old mode 100644 new mode 100755 diff --git a/public/footer.png b/public/footer.png new file mode 100755 index 0000000000000000000000000000000000000000..245380a345ed7a818520c067b1d58e1251caccdb GIT binary patch literal 3549 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 S!GY}qg05V)G^sIidhl;|l3XnS literal 0 HcmV?d00001 diff --git a/public/header.png b/public/header.png new file mode 100755 index 0000000000000000000000000000000000000000..d0f2634e3021d51fd439a2c8db4ffd574889b9be GIT binary patch literal 72302 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⪙w9>*n;Nb zY8c1mC)@Az4rGBS&U(b{{H$kev7B|EPH~Z4uRpk&E~fa}hHK-hnuP^RTM1r5PPhgT zU+wu8;dD4>$zw$Nc$_y2(^&TBt{QEiWgov~1t*>r(&Z~#^Pd8MRb9@erbK3MW;zdR)$tDwYhc%G0GKJyE3lnMEYHpa(pJ$f_^$r&{6c?O^!hC&8HAvl-%pz~^x=)0{%#8;_fxU|f77?zOs57eu zzMjowtIrp0?}>zWoR)S9{K6b8mYX3MyWG#8pJ?-(^NP#vI}Tze$o4 z8Po7URaI+k^&3|G76_HKh`5j+w4SM_n2{fO@Mzjn5RUz2#aq5arm{8Eey|wqPuhme z8M@*2mK8b(%x>ap0vGyoQbG9CCO8jE63SBHcQ4+vySl5H);{btC=iBw={5G;AYY-U+p#&d$Zd7I2G7 zR7Hyw)l_R`JfB?Rw!5xS+B$E6O?P*|S|W-v8>};UZSUcWu!CkQ8Icd?Z3SRHg62q*Y{W8J&N;@-T{?990`B+*JG12e>VQNNcr+#BWC(T4FXcP9`if@(*T|{y{ApU z%3az&>PUtYuN!CYN4`tB^4wz!neESPDcO8>_t$WHl?@WQ7b}9iAf&c86db3r z7Uk1(;bY2XnBQutFH3TJQ5w_1E`fRu-;#d5p|;<-Kh{%lCTT7@DD1wPX$45`gO9N6 z0A9aL&!=;;XE`@L<;c3VA%~VqYQSDx2Ff-is%|07C9V;28nMOLQ;xn#_0n~r~jn-@8tJ2NZLn`Wt z1d5-0dJY~ck$(N5&Oa(L1)1@^bv_QkJ(J!<%Z{v?u!;D2o+~%^Ja6-sv)J0D_3Ko< z!wJn;OEaTl51ZwykcG_J@D$M)zAJ@rEw6ja`5XEfM<2{4Ea1+NG}Q}5><*Q}peAe_ zr(K5Ce~!~0o7x4G)|RflU7f){;X8EmbIDSZp18K8-G{Xe=*p(-QGcBam5Jr1A7`-> zVCrA;;8Ee_xd-GgCJ~SPfN6&ov&y(pG(NkVKoYy54`WBX*z zKVL@pwZh>5E0z1k&Qbp=`%Qt%M$H#R%CC zCRGne3Z)1zi|pizw?fIJG3Yr9%*?oz2_h_eSPPws+v!8v zHGDR#sLWLqV>Qi@al?8@wTnAxkqCI)Qat z3`D$mML2rf$j>SLZODq|e)@&rdY_zEGP5t|Jmn*iSl&r&#OJ*9qmA*jb2b*j`w&Xs zR<<<`m@!pHim~(GxiwFtzT;=~Us|-q=#te$3Q0KYCT$_El_L7V?@BF%>gQ8$;-#jC z)n8-_t_@4pZ(_Hcv3C7y_@u)EgC=;ssVFWvSP1rz`leH&!{<7Kjt{-)v9!;V$btr7 z?*u20p^0O-6r_D0zO{%xfeQn<;I%uyP2hFdCS4;omPbBIJ@M!&?`i4NO}NDRwm{bA z_^0;2dXFt`O)J{;O5oz_BR6!cLY3X;D&n(%k%ZTq)~MU1GpxWtVhLmVo{8oF!LvaJ zV*_rQr>{oSjO!CR5qWnsan*|7`F*0@5uV?Qx1C#D&ur$z%N#3};M;725S>;IOmVH_ zA{*mzu3prfXFVa_j7^`y1 zx3fUrLkV>Lt=lIaAu_3e^IGklyC*oxR^5a-dG;GN=G5n1EG#&}`rr2)-J`>reBCq7 zXGJV*5V+A`u6s*H{g-pOQ#PB#hsqBXWcLu@Y*UBd&cdH}uqMy6meo~M1jvS+(|eLL zSJdBxTu;L8T*>2hkfzrge9ovLHmkPhA}wvYG`xT4kc_&6 z8>KcqMYapgM?8H5ZQF4oZk4tq8x-yTg}x}K(<6j5Q6wf2(IWnww$6+S^0^%EMNGn$KV4w`>!60sVR@nMnY^~TqqAo9 znXTIod*pbrzVsI`FQ5R2x|_y$?m1b-E$p6Wpu$sezfo5hr!Gywd2y zJ2Ok`IhRfp(R1LUYBaMJT}2)w8R-}IlqqOmEDIyR*p#@i}p$|ux=mMwNt z;AE2{U2pH|qZ>noB%;i_Fe7$2uA-o&qo+#$9{t15{uUPyD+5yjnG|cyBDlEX!WOVV zn;43i82V_x^U;1W@s!#Rm&hU=i9qlSjqPB==anLxE(9+5PLXOE(DO41BMRaP0`Huz zXQt15Be<-t(PfX7*jC%@{Vuhf(1%9xiHGx_lT!|P*-Qv{J{?VLu=q}1T;14Fj=1>T z;*-Ba%1w=QH*RVhRIX2t#L?~1&!e|IJmifH%V`;08>ybi~Kt7`@f(WG2Zk%1PIav7Z78D0;1EWGV zlw-r>YR&IiWur+cwv>H8v_j$D@|Pz0pWCuwSJ?lO6p4X&SL(&6`gm$@im{8}y1ym0 zK95N7cM&yM&W8{A-rSuqxYmx10@Y@+q8!o?{bU}W+s)Y6wOx;#m(k8K7ZTCD*`ae~*)3X4F zlrGIUrVoKVtD>rE@OX6Ybi&Yvy_UlWS9fRxVc2BmBx8 zLo*4#Hw*0#L)x4X?rMTz-j(%OLv9NDR_Z$LGjU0Q&K`lFebdG-#$kjh;#+HmGYx#A zx%|fxR?!$33U{wj=$BKjo3=9Z=8g5|+o!{`C_AL3*s~co&%B^aQ`FMkC!_Ys>1~rb zH5{z)pBhLD*M9PNEDyi&h~+7&YGGmW(XKAzI&j*VxA5z?$X@BQLF$Y)aMXWh0jIwh z8R^unjHzt`qWVKWd)}`o<@yWa(|-{bmGkeVry-SQ5p-P83#r%Ng?&t(?R4~huplm= zx#T~wgv4l#00!vm{%Eg5rvbV07DgO`fvB8)#*J#yg9HKRnLwpOW9So4&lLwz2}Q69 zy}g)4{9@%2j-!dQUp5bV{6u&I?Dn#1n0$y+XhE^m)#t;-5G+=oPvh+L3%E#Ihkf-< z-sE@k(^bZ!SaCG-j!kUSoh8lf(9-Sx%ZNSW96Y6vbD3MgybYq$7Mg3&7nEhfSUQzq zJz~!ouxI3I_=B!xy-CF4y$O-sDUl3)J(P$O@p>scJ_*i-nXYZpvWHZ;|qQuEVm8=eAtDTT#Rcy60v zfQr{cIj!qq7_2N&M2Azk;q z#G{ZInCA)1G0h{>ixCi1qJb~3Rjx0tdajvEe+C2T)+ROUx9;TT>lS#EK5|DPLP^-_ z-}nc|O;K;2BI*~ODDF4m7lC!h*eUw9`>Af7;<=N&Hk9}H1c~Mut!-~3^Zz9|fy=?5_28nkKtNnoMRVsT;ERi9IzU|H3Aiz|*sSH}}dm z&jDALibr)e1!t|Uw8m-!75@i8?Vq6TyNueN&mcSYXM_Y({1+6`iQ zDJm8F{|6cOmI$dLyFT8;j&%hWbT{oBy1_oqKI23M-7btE?v*jr!694niAuP2M$mG? zJj~AO>5~B@!SQ;|aALyud9mn#7dXzM-O(*a(fPFD$picFBK0J~=Rj=Sfk6n<#tW)+ ztc%*1NJJwDQt!9wtQ;HVL21{`i*W)(rw|`QB^j=KF=wDot zQN3AqHn4;3hsQ}_GZ4@m-Y68SuaP8;u`ZZUQJOW-MJIg}G4G_&*?}KIFM44-1`nP( z{wc1015W+fs93WQVRdTu0j=t5$JHDa-4TY3R6KGysVxwn2Gxi_}%nbwBRTG*!T4^mjWQDHG|TVAdjl}=Vp%*_Ct|%b+N11 z?=-J7r_0X5iJN&7L1TE$nB zM0GFq(8_$hV4UBj#a?isZ!$75-DzuAXVp$4acf zXgX))T=H1A#ReLOn%j!|hsXTKr~D+dXB) zYUKzCeN|nkDiynOfAuPEwA#qHUpvc)Y)5$QuJB267bzt4sETo;85Q>E!nQaKkx zZt6wCPZ^#h|YmwhQ#HL>$49=}V(@9vBB2iSC_*rxUeLK}{@XJOVR-C*FB902q#QFkvy z{({Iqz=}=55V({^SW+kHZ!vS=-(tqz_|?do^TZiTKZRc>ez?q4=tv5%F2z}JWP)DcO9Y7h*okdJfQwN67ql7d|Mt^<7#DTd5_u zzN)UruYSUuXKjFgLjSw#@A7CGMt*1wt5&qMF(q3@J^H<*r093lDl`I+DpMdy&II@f zGNzhiDlb^K&Auv_0#pM6Qgj#bC7gd{QDG9>yG5d%(i9B0*77LjJG87}mt?SEQS7pi zSb*>CCsLI3_Q5VWLysOr#58J)m@}=l!|@Ht53vRcE!7?rLc*?Zk6C`GHGf9$TU({6 z7SreNw~F(En>1!>COqeAS7VDA{EqIQq`q0u*2c(NB-zCL%fV~3cmhlV)UrDhboJVX zz$LQ>USF}vt8v3fORACfbz zD)xf>YYncE617e?>_s8{%BHYOb}PVMYOFpOas>dzu%5`-3Zme~Hwm~y!rc{S<502SOD||{^ayPOJV{EbqHRh4*|6*J zGpIw;?ZUbetzm`UvHNB3NWI`i(jOL)sBw(of3<6{6sB7SiEw$Bxj7s0$Ia4E*wf z>$1T5L(GU(@G_^*U*wAYkNRSFqn}Rc>B=~J*-TGF8y0T*`GeQ+>5KMg4XIGTn`~AE zfMHVs!~ELfIXq(m&zNNm^pVy!4D(3ClO>^w3-0@UH2D%Hef76pH#n3=+qHI1&}3)) zy0<14Ng7XDsA|!qSV#S3_@xS3sCTGj%bEr5sRg`y>6`YFIk|m%-DO5oPOuX9pg>Db zzC{oB-7_b3`AYN+a_%)R2<}viT7th&cQ5qfAfq%XV)-6;ObWo>^Wh5`xOl#)tyamk<$>yD-AtupclKLL?{6QYN}MEeD@X#kQ_h?SGXP zd^I!9qB`dRqqJ9b=;ffritGDCr-Y}``4VAz3a)n0qH*3LWoyw!RD&E}gDUFb51x3` z!m%;$4Ou%CGGu-lv2zc;h#3aeWrno0^icWaV`6`f?u~6~*=DO8%vY9!Ym+A*zjhk` z688HI%t%KU4#HP2GFPVKL|BB@GsC}#{b0ynW;`ON6&qyH8s(GiFalM+?yMTIvkuO+ z*xZO0H}TWY->VPoHy$uV=#3(rmJb*FHGcxa56Qrh7&s=aIJjX+?$hgeL1bRzrVs9a zum`9tYt1Shyb!X%RNia4mHWUyX2T+oXCl!F4zH%SrrBxk{M7_{O}HkyH|^Ub?RWc9 ztTZ)3n{G3#QORQnRpz~5CE7f}x#Fv(%qyEa=}1$R%ucKM5hl`n>L|CT$_q1!pLRt{ zvKhuC>&nI}n6%|%i*Ee#qvBw15J~eJR5w3usPFM_G@`2@_-rlO2sUkzOe&E$RiPJM z&#uo}@;$QuHcF^brv$#9{V|P6I_##av+yeanjII}u0IMza@n7%aR02`CJB!K4czSP zkm*_7uW>AQO7-Q{Xa?%hzaAP1bFJy`!ohUJ=j{iD`ZmdvcdX#y`AygosI%+FRMxszzBD#nOAQ^hYQ8eZXT=3={%I z;#N#B{hX2w#+Gi!zAq{yU&>VT|8$>_FE+8=yGZpQNL*Vl`L6Vb*PYD%x2B^tg(m#b6T+K2fvSDu1pB|^lptQ?xjT>zu@>b)PUM{DuKp-w|#MK@3n4jk=CS8X}Zdd zi>LLuwBf-@(}s6GNw9~Q>{YJ{3WuJ{42gt$4GnpAzMbKp4$aIMd_*5^=l}6FRmicN zBwa;Xf@~;lw#M<67NwO;b+ZZ$+C~y=Jr7A7F4fg(ezdDuL>#Rw&<h42z_(BTzHAX^?_7mKG6x>H3&;Y_ zB8%&7>t|5McivrKb%>h?D+}NA=5}**W1LM_Sd9^Y=#cxPVc2XjwfxJ+~S6d!B~ z(H>+)z;&GB!f2hkTTtj`m+NSK0M8?Am8;jiM?_b-vL!myTW=r6&2I0UluStO%LWPI ziv7vMZw8GSp*o!!`|7BwYuI;Zjw44hdwK00wNdZC3u53Z!MsRL+9xY~YL^vL#xiJ0 zJvQPw2}@#x(;F~U>h1_AH`BOFhr|M@FJI{Hj$AyG|e@l)y#cHri4W10Y#-!paNOcI-QV9|r0 zR#90Ad5Y=9wisB1>D@6K68WI^R-?)662`n+V$4viMx~=|U>pvqdsC2#gfi}6^*AmbfluJ# zD_}_pg27D{OI89ih>1U05|~)MhKk*>6m#9f4rzl80>%wH3CjF-AyZLn!G8fP^1r2@ ziZ7XY(p3E6g6e%;S99L}u5VvTj3PO+>tJ9ei=4SXnpGl!6j~}f@`ESYxx?Vs*htCO z5_xI(5`ztoiv-V2zu#{{Pn%Adn{;>!r3C`~HNo z(cZ|!^2tdNk-ltA>s_`8K-*T3uK)vCq2J&buxC`flk0wdq13@emSTNN;gFw?@$bqv(p`-Nj)^?URT|MW7+);U9}gxuy1#2VRE~`HQyW^h%qH^?k+Jc_^>^XYQ-JAZL!gHV` zEqxkGtfdQWts)tt$OvCRrb}DOFO(;j1Z^xnkT4@*c9$?P(W7UhM3r768lOeqhv)MM z^<-SD$-Hpxc$>{%w)#KfW}NSP$zpt}3&bI1k$v%Plkx40^Es~ol#!WPc>k=Qot7(f z@W`o!a)m%|Kx0Xdm|noLgubW9_gtG0_SZU)tWA~!_ zLhU%wKKABsKPMNQc_H+DOAC8OnBa)}x*YFUThp}dN z;*P0%r2?|Z?9e%(nBE&VRfPc^iII3-v7>4-To+22|A=9L4=p!$qZ;rv8yPjJNi>vw z_fs3@TWt6z3=r4K?m7Ld#FWuFQb?t|&0~=of1Zn$1Xdhc{)1u5So;vzueTY%`QI>Ue`(Zs~a2uf+mB?eO|tOGpcq@nT!BkPE&{6Ixnt zdSY8|O(qOdXQ=3Gv+FY6yY-6n;oO6|;GzfXyt|}L1MO)v82U3A$AR_=G2G)_JNi|U z#D$xfH6cvt@n57H+XA39J9J)K7xxB_(2=yvvV55C(Za#CVtpR-J#F|8T`hm~KcsiX ztwwgrTqgXLv{Q}+TkH{I1L;H&?c#yKW=~{;vhz!a{tL@qhlpkA@+`swv@pb-JP`T^ zi~7y@KYj#HuqU%h+3`|pvyN5Nfrl%8VKrJG~d&DrZPQ{?wtdXXu z_#P?-y)o&2x*EkfZ4oX)*UWQKP0}%vnOS2zk9ZMVZ0!`kVKCJacWv_sNg2D=;H})o zALfy*V$?0x5;OinC&3mHIysEqu%&N8n43s6&{#1y=;N~9z!OgzN?oz-%x!*0emjeg zh23aC9eX6wg^s>;;laniQn_e@3mD5NlNU2z=Ay^pe>FHYZYw*?4z1Pc7?Uju6-84w z`P9ze)*T`u$uBn(k@*l6PxYPdjzA4;jYiE(kFSEVe}Xs4<&I29QFfY2GlGeuq6!eV z>pgJmJ#Z?X{!Q{8L;e0#3VEXwEI~Dk4Vz%;HUO<&Qv)JJm4t5b0bIVThekuay}e0* z7af8P#AvPOPzhxR2?BKMDL48BC}d%SO0OK;uyFH&*mazecB_b=ku`Z?W-B`b^*lxC z>IEE6*eR#+F@Ac8*OA=d>jf9#!X6DjrDiL-`np}dKXSu`)6j+vPT7a152T0U;}HeJ zn;K;cn{CtxJe5JU-)iE!=rvKd(JND9i=A#tkP#ifYf)@oaT)Ki z6@0msoHx6wl4`I7#&K;-5NzrMlsyZAb^l?d=$ z>z*7vtqntIhpjK1d3mi?!1))-cc$%99B1=S6K&ajOC_Qo9SD7?GCGCJxCU#Z+&34U zcNgs$1&H+6#Jgs;t-CT$6q zk#oZ8LYM&{PlnDPH~uof9JF~3%zZy%HivdcndDk*#0q(OdisaoIi{8wnNncsmA`pn(h zHhz65?A?JHtvL3GlLFC;wWd1f&U3zn-Rpuo-ur~&Y65mc+hHe#j<0wm4nNe>m87OEegz_wV zYnR?&kwlz}{WNKuT<02kG z=~bB;n#^^Mjt=l;MCoitkY<%gyE0O4@RR)rsg>Q{RY{>un?Mtw z4U;)zqr>E6wyroe%?@rkG3Nz#NkN*M+AqW?I5>(=H=wegdg4G7=+Z&nS4OPoi7*J- z>$=+Ol-iyRprjVst}dylsj0cpD9gbi%_=TfAmW%$a>pt_Q|lmJR=Bkezf*(GdR43#l)zzRVd2f-|snB zn<)|fC+;QHuq~}^W$UR>D`G|bhVM;ZpFD2f6df9RK{ZZS$tg)RSp2@`an6sa|*=l&jBGkBn%rMCiGqy}vShA6aMZ{9@6014LS zcG0Oa6~BY~Nnm{g!N%W-czrN+zZG=OE!iT!-{=a~v84nsd{MUrzjm0u^9L>C3we{K zBlrLLL+GU+{I^|qC{HCaK1B|-s-xg&{Mgj}r2XTv5_uUJFKJ;;OE)PX?qb#c*BtCX z-~X3wU4RE@2CH1Le%k;-(|_}tXuJd!SPYoDyI|Avf@d1!N>et${pfe{pwAqwfb6w3 z#N4rGIfN{Gi&EN=Fnb=#BB@~3tz?{Y(B|Do%3g=+BP)Lp(&|+&qrZ2y5utEkw`8R~ z^~a0D%zVrSHUY=UaQ=~{=HJ#OBNgz<%<+a-hfB1ydGUfB6R6^3J3`)Bssq{cB_S(G zux$3G{BHqqCgbwN@$AwOvgz=4vo^h1n@VNw;!W^mSv*XrR?Dhwl8&Pd(KzD}U!U??4J%B?U;vkS=#}QuuUtiLd8> zwSXC}?p@FnmUEHV+ts}Q%4~|Q0`^15UvP;H5D8gpmi%Ge0mU)0B!7x=&P1s4lkyJy zzkXWHJ3`czUhk6RU|Y7`me>j8xL-0baD_y%v*U_5Uk;NMx>4^XL)<}Oa~mIn zdc+8CQvWe^jDty`?e0hdk(#V$@EiT8DA2s0g)$s3^l@8P%>`A@Q6$L}^|E6@P#kG+ zw*kDth@EWGb~ZB{JlE=0ZSx^M?TXH{RU=ua+GAY33WTf2#lla^`ih-@d;YH|{wcdIy zNT1$GtPf+|gt;E4y~eA(mQl-?OHEA`+8%2-k3AhqT-XIcq>yfZNsgoBWy@|1Zz%Qp z&2-nC9P{g?G6oAmD_Dd!9t4vhD*Q*EB?-f0J7*2H!UwKuT12RkV$ zjE=BSVIV1_ZhwEergbWw`v3W9YP<1QiV<=~rsqZM!Dv2#lWy4--ArqI*9HpVWzFy7 z_T;R+Y;<3-H9w0ko7&k}YLyS0YKnq+co=zm9H{2eMV`76;I1ZQ|EAm`XSTokxPei@ zH#o4kSg(ey9)z!wX5HirBaunYJho_;9B8k?Hyu zfmoVB?mypoPK6`NLW0ydoZy6ZIK7>j0QhXh;}Et`Ip>#CW+(A!6y;ROW`Kge>izyn zSLX*eA1F3yhN6*-BOt}lYSD~m;lAbz8(6g8EgYv-h2`O_iW&p(L`2Qc zxg%ExDJx_f11K;mc)YUh>Vt@8$|V#P$FYn7h}X%{&c)4$=MnGts@g@W^`Za?XeMGM`rJDW@om+B z>F%E7KS%S}KAtNXgH#fx?N+w=_zMxFg=uP|URpgIO{OrZIVt z%W)j$l(~%Udb?L0$=&##F{_E$PI=A8li%K(8u1(#S9F&$N7@ln5A6WvL}Gmy_-e)^ z+f2mh2$qTyFRSPp?JZ(BPUN~DVqJ!hk^fGct;Dq%Q+#}VeQQD~2DPOnB`)jk=`|;Q zoYd;n&KP*&QCk#fvzaKhz8U)$(K z?V2=d{r!2sS2FB(yW!iq@EHg_!5qyP!!Ynz(N>tJoM1-p`{f2fs|t-%^P!x2`^x8Z zRt>zM7uiqQiXAt(RMC#s%L_Sm)7~t2v)D!jj z)}X~3viK#x>ViY0pORFJuJZ6oPy5L2dT;0oo;P&t<O>b#CEnj!3 z&LMg$l#z(7&`Sied@HA;ih|XELui!y{k7F zLCR7AU$rr@d^kBfbpEX!j&rBzol2z0D@LLx-tAl(Q%i}3LH{qrp41+)_E)(<%eAh~ z?c!GxQ@p>=Rly4Qb#*mW94AE&EOvr4D^8Yh`3e|St$m{(U3$?rgxo3FZM+;VmG`L2 zF|y@=(m$Ic!&WV90*-UP&zQW>gl>9u8d$o#Sn;?1DWhCC2mR^w%(j6cPfnY~Npj^<|3K+M9# zB<&RK1Wh<$87IF&v8eD9Zwbo%Hc8NL7tG)@j#kTgX7yC}rrl{$ejB|_p_iAF9x|i8 zhC6YGv$Ss6Y8f%`yWz53#KwRUoC4YH<)e_M!qJ%ARl99XIReDY0h1;!=6Hhmk@;dB(DtNdCtmL+_)8i-h(Y^(%j=7n7|u! z$s_g(Qqb7BPb2+dD+ghd2Vsk+Ng?fW%+;gJ)!=Xznn8^BaG&O7gk8&tZ;pESNFbA_ z)+|)jaBX~Oh~4%$7vAg*pR`r>VYIE>UcG(c$L|-?QM+Es_F-*02M((X3!aUd4?Woe z1|4Z@ZCX%=Z_kcE`*xZd84_CSR zjkCO)DC^bzuqiU?9*>rGBe0|BmF;7m*z)@yD$|M zmu8oEM8p-Z|6{?zh~{M_?@cdtZ_~|`*`>u394GC~)8BkRw4IrOk~BKfwwOUPpvgXL_Eb?$WlDaRus#BiyOhY}>-ejwYKF zt6E*UqAI7b{y74Z{^qj}UyDbVhuenot~V6TyQP&MlJ9+* zhVl+TrRhH#uxT|$4|b=Vg%hrbkX3Bs>7c>m5{=pbRgCedxRkSR5sII2v25W}vXL&% z7@qmcOcJz1MDUeGtQ`+76kYcKHxe{&pXSdcr`L~}Izdx#E?`Yxc=HQ5-;8Z8e zJ14Gu!AmK>_S+CB>PwrdXvxBZDJ_NPt~1Nxahzb`j$imP>skYDil_0V0Ic8N`nWP+ zZl5~&ew_0>2Z!!S{*z@7bD)ZB$a#X9glfmkwaz@3J$-Wz407I_WX@seZxAdk=YV!c zAd@4D8HHKpOVp+~BPy62k+?zAH~_o({i=w+Qj4i6Zc_Ua`J{yiqm&fWtA8VtfBxCy z7f$h{v(Y!-Nm{h*w)v6&pZ4A}tf{T*_ug9tks?SB2nfA0O`G`KbOt>VLFhF-|_lD zsB{J6>IG;k&_FcD5fIV!I9#@}NtN~m$UFiaNFK6kRvmu*G~<#}-!6~uC(^gf5t350 zYA*nTOXSAd5N73N60?i^l%aL1^0y=&V7?sWJEcM#e?k8KlDPLrzoV(-1SI}wC}%#E zFdM!5asW7MJQ1)q^z0Rdf&cJrJK~IYf;lAlLV3v(Q$mSnI3uT6d%2urqUm+%z7y^x zW_nxODTbf*(2YQBp`hW|EXq13?8Xkq^m@q&RBM}f zXR&)owlx?h$8R9& z^zlBc{yv5C!L&g6Z4($W(^^JX_Fm8I_&^w!OSaTaI7n}CqWN^;)uF6Lu6ypCGEpAHlh(bgv@mtTp%+{=kb5 zj?;S~Mkuq(z7HOS+~P^I|3 z4=_7e$>>EAhg>6H3=`T;vmq}%2PRzEj<%=0SrT}A#>D-{N>@uww*2FX<-c8jnI~V( zKY`I#Zr4IV?l*}tD6K!22Y}~pEgx)Gk8;mxambwlvO_Otmd>J%uJJwPPSLf%L%61Szo=@kKx&l)Q?y@lf7Op9TSW9BDrwAe9kv=s?Zj@26B4ZEDxv z@u&!KBclFdzg)bjiF>)N=kdnn9#P0%SRg-X2y8GQ;`TdoHd||LlC^JhsIWkI(q&fU z1i9jYpCJ3-dRf5C+CEpx({h|p^q?WXjQprxXNxSxsUNHSrLvI^Y8%D%o`rBe*PZAz zO$Ix#A$OQRaRP>Ua6!VTYsC$HxtaMXun@2H^ECU`%aZp&Z91GK%4gr1g3Wo(*^XGZ z4B%~vNYkWnOd)X=LVLU*u(9N$8dr(OR!omsi>`_xGgf?b*s8TIzi|hd#1h4Z-dFWB z{&qfjIU}LwE;+FE<%PKypPNe1H=a?{Kyio5gd3Jz;pq0t-#U0yEgnf`lcKS>`gEQz zbErP_``CM!8)&1xk5g6hVW#>QEoEJ6r>&dDB_E%buZ(V{$p6fq6C3TVql*&Sv~Fp4 zua`fj)b0-!_9tzOHgCJb7OEeNzxHKugO@+_26S;)z@K5-jprfFPjmYD;xpuz`gl*E z<}!PmQiO|M#2!JN9ddcP`;%$BW@NJtG$eE$^#NpB-v$E9hIWC%674Q%O?IeE3mU4)>Zmp-g3yf02+Hm9ywtH#+N9>O`QZnt4WkG~?vmDi zfviDK@KD?)(t})B1!5Ku%C?dmXRdJUfZ#{6Z>fCN#Oa!cuNtNtp6YtAyA=@PPpZ2H z#l^MK4X9>K#&9bn3K|F0LdZ$-9psR#@w$S2XZHo4ppRVq!|oP)0FbfN)R)VfNbn#= zYi2?WE2XFCGPr=$?A4qZJ2?PHJPbYfC(rAZ408i3uM#FUyyM|}0!N@0$!IWWLYy<3 z_>^9^A&(EE`qh3bXO%rAfehwGeMj#5tAS7A9kQ;Bk7thfl5&{m_vP2Ivs%$mfLcqc zQe_?jDY|SuTS3ql*MCXPv0PPDCI97n_!9h?(6GW1SZ^B-`paSqsCfN@9M5t$bU!_U zMCi6X3`YAJT^5V?inUT^$@<6c{H$4vf9>J5*IVIjB1JtuewC}p zq8}MyJTeX#1vLH^hPr;)^!D>C-{Q?fA`Z2}1rFR_3^P zN=tte*{W6yGqoK=4BUT$yG!e zmfv3!pB|`el%#azy(9N$5>ceHzjKeZX5H7)OvWgId1V5EzgGSmrxrxVKr59>f7miFLO#e{s7I&CHp$^ z>J)eP0P_K%c)=)jv7r&L|M|vE3R>uShW3dd`bEXoK#De@-oES0Hj^uQh2M_jP5j>( zh6Tc=M@Zhtcv%^JO1G(m4O+>i;Js$y27cFjKFlUZ#e)fB5#zK`56<8I=I?vgS&oXrW_>~XDPh>8Rs$>1x0=E+$?4Cf ze9y*{$*MZ!6}2G zJRxu7qsZe6>$Gl=v2`Z{AeI;1ZL)tiUQ%`j^6ALmOfm?ueOL}lg>riq!drj52z*6X zFM;QAdWI{ z&Tuoe=Jo$)_(gHg#b%IJ2iuFe;#@+4rxluz193gZ*jA`+)$yka1-WOe<_mh%&sU1> zky82zQ|7k3!SgfS2C^31cVx4P>B?7`i44#!+^o9K&P?U6-I654eDX=bvHX1>ZU5{^ z>gKHeRNnb8>1jsFn7}ye7+a*?l3L6#4GqYpV$rsOB{<6k#LnKI&P*!fa8MUoU>_k48g zf6qLB>wR7(Ghv2qw1+r($GgRkg&MVLx3X*=39;_##=#Iiq9bWR*ynGT>+L5KGD?Yv zLvcGSLcbQP>wiQwq+!$CY;63N2p(v`-X8-&Scj#Ak87cR;5?^3Jvj9H#D@$c zU&BJio3?C-v%EkOcCm2*mG02?F6idc2Oy5KCetf)3FNpm@NN9~6W@GY9sWQjUR<#H zfV0P5GG$h4dIQjScox#AN zWZcZjl*YQKIh9QnjP3bx^SZ)nDqdqup zg2-mbMDyu=G&AQRkEbcg-WEUaami#!Yf>Nycdp)5dN?-A1)Z8<`Pa&uft&8}fB@+qKml`6y!~1Ai$x+3Q3$>1GrJkCpaU$JMyxB?0&& zlAf8A8*SJ}4+3(yS=GS;j@W=(NeMY-gV3|tQ`6)t(@}F#Qh~d z(YD`D%U2%jJ8CZ_ou3bMbiB@Pb2e!;-yE;@%XE-)yD|V`TsDED9bq$EwtBD`tu#|i z)tR2$X(1E(5!Pt*C(_FZjV(RP3QIR(Yk$I*F<7?xNI#35ky7e;F=Dg{?(dbEV+2R5 zif=X-4<1h}IM>3q8nb5o`HmrLJhiSs_|N=-wXvDv0v=t$2w0KFFL=ClWN=D1WI?6v z$h6Gk(@5MVr^hGnL&;|So-)A=Bfs?V;N;eorh&|@fU&uCvy!XLaQXL6Q~apVv+Q2j z*3-9z9(mTokaGTRgmrI@WbPH&f z3`^{xnmBM6Bk_9(C@@k;iV1Uc0qW9HwPa3IMFJ3(@-+Sg$D5hYvu(<+$7{kbPqZRYlOwtP%B_zrGbIamHvq5_twT_ezgM3x{QR0pDboP@Ui(w84bTMa zF*b6bIC5s-_}!WeHlB9vvwRU9yqLCSIN_sz3TugmuA8EB{$l_h85+QZo?7i%otMyI zCGOkm&K^yS)yJkS#6K?~iNnLkA{T7$kR{T{p|_@bUGsybb7AIdD(AeGGHVAmV^frT z%I37-7=`JFirl$ApMqCh1IFr|kCO~?S8bI}j|J9`HRaepc`81kb!f*cjTxb6} ze#+xzcmJ%5cza?j$Kgtc${x!7SdF(vi^>-|gIrm)wHqP3 z302LBK7ABtd3#GhE(n?6Z%Xdwc>5rKIql8pffNAci4RK?8qI0H!N{o^g^T|oS+Vj5tT?^HdLA%jfrY|bk0x$}(;st%B>=V+yr6*QZ(N%GKB9UQjpmZn9{K+!Cn=}wa@XaDmIk{au;v;8(- z2mgUn?Xcv~$St<{~EL?^{0Zg(G}a?KyqJ@>nlUuiPAiQsH!`^ zG^7J>V(gJ4c;PaAV3O*Rkd3t*kD;K+9n~Qg#j~x~S_j#6L0%_US8!*RP05GUZ*!WhEFc`MP zHHrWf3)L4xSiThkax9J`)ry6Nm1mgPYT>{xXpZH~C~LvTOYl!kmlEZ`-1SNCtJF1yo1mB8fvY zD4)?(T1|I@@iC20(i2Y>CaU`VbPOf^UD&xQI<$)z>))YPSsP!<;B94OYl%1I#M~68 z_J*%L{*D5l4B$pu(VSCP|eBl%7wssC+A!2{OU>Pw(sJErh!za^`_ssgU~ zA{aQ#$Q@~=$MCq|8sPI9alV}W4Sj&#w;O)0QUFNItB)ir_k2q4H{}{{j!}+80;)x??#2S zpK4;TgL!cu%V_AtOgT8?JpLUX~HEpY(q;5x|Wt^a*jDkNNo0<>bL)i&Dxp7=-hd4RHvjTj)TTcnUNaD`Uc zy0u5^P=gQ^_RGAfQ$NS`uRZ@d@sLejG0f2npW|%z{mx@oV6L>dVY`<`a!@{k7lp zmM;v=0}iKN27ye|w8JqWW<}B`!St(97Rd9)iHu^wIkRdZ>#z2;j()_YG2r9&ym#!$ zhb($AuXWd0Cboo8F*E&#Vs!=nPP>6u#mCB+HT9jz%9x1wVG|i75V5W!BY)&0a^+p- z|6tuj*^B=b>|3KO>%^N;wi^c2wEkgN#iw!uhmVshEDm+>qQ6pwsTQMrj@OyR`GTf+lL(KbCk9XxQ1NiJ%<1(_6}R_kkO4<8zXc8Sl7-iae2a0Mxj`X~#%qAS z_`}-?z4h*gZoSpRmnrwjWT8yVo_fN$B8ASDUj_TIkm-mdiba z$4{9&=nSiDGDbI(1j-hezRhqz3>x!=CNeA>cBW~qW?#(9wXcp0r1>tjF>Pry+B0$< zn2ykKHYMV--KFq;e>%@$&rj7t8R|F&N_Y4&9^k5qZoN;`un^O z2;u`9B_D~qL*MNCIyz1=yVFi3#gLd*Pf8m7+C{;krLdV9uH5R%=jGI;q; z;?z|!?&t0caWibwo#(3g?4QkNwXNUNERdyUgQ8jn6B?$t7Ji+atw8Z6cyOwHxS!2k zv6mkrJI^egd#%OmAMZ2-6D`xuoky87)%h&2_Q!n579bFRZGn(Ck7wL`FEP6#K&p(k~4iZ zXXh$X#$wx!1&?2+_pxM>JNke{JE3bySF9`YAj~7z`-or0a`6G)A7eR4JIU3{yk82E zQj~+=z|SE&`x%uMrIb=O!8*`JGl+{!LL})EOJL)ls207#f*M^IiGPaad3pc~1%x7= z6$>7yW(nPJ?Di9(Okr&iqBaPvDr>5@p3g{Ee1rDsLk18fSXZ7p6Bfrc`|up^1Q!)Z z>FD=u8PP1al^&G)-B$_mH0!PNUyvtC5?ls$VD(l0d_?@Kf*;&gvR|Ff0>%HMz z1_-ntlB&0YsF@VXV)eaBa z1oAzH_k<$N*od~6TGoRLtv$pM;xMOGnQOwK`LPdHGlJMjo+$u2FIb)P!(rF2x*3~mLHGK`8>qY7n$X+LDYkIFwduDGr*w93CQ1LA41gkg>vsQx%zlLf zI0m;1)TE+8XeWV~zf9a*f`F85dgN#Hl&ol)o0wiXzS$_$)=%D$AXBzgWYU&r-Z`l` zattG%tUrphN8m0Z>-jQSr=+g&DkfD2{Jk3 zc2VyWd+rb%Ff!aZm7*_M=?pr`@!u@^V)9`UIXe}UHSrmhHfFUvA*|;~iweht4t5F( z-&O?0v@Fl`!l7T|3p&GXMTi3peU5;SR;OM`Qizo*K+dU4x1 z8%SP!%)E_m0TvT;S(9gfNkS=glnMbpfoY0pEcugyd#2+l8FY((;I3SMlAE8b)Tr;m zhZD65Y5YWVtvA$+%4S^NXp&PHKuRus%id@UwYUEuze_opzaLnvoTpl4!G;aGz#`cc zh9U=v$corg;XSwrVO<()cRmwY@H99epRQLFe4c*K0_y3Q`|hT7+p=GJ)>Zp^@)_iq zw0?^IDbKcsL0CL9-R`%nJ(;4q6j>bCT>XJHQ=nL(cqe_Q-n8&?=2T;eJ*B7vs`+t@ zO1^1V3=s_<@eipvTEx`7nAZ`*wm~ue((K$^wyuKqIY{B)|d zI{wZBj?Ux~puX<^x8XD9FB09M^Qxumix z#XFwfnZ^6&sY?rIn}}=dSaVQFnMYu6dQ3r1aMQTKmPReg?hU5nJj!{-RF}DT)i1CB z=7d3IWK=7cH6yMAn~@-_WYYdjHD6BeR9!|65K!b^9z^LVKXF;xxP583RJ4uTCIxUT zoJgeyI(6qk1?NE=J`cb@tmlnPWBUYt0M7qVF7vyfx4(7X$#T)Z)`t^qLG}-2l}o+! z(`1k7)E{uw#686?n@L{O9C`3Yn;JAAY*Od6uYhz91R7`p9iKU+Uqv?P+gVz|3`_&j z&SQ4SYAr+MkMG?KaW$<2^Tb0^(BNC;HVU@FN?X|n66;T_y26NP@`WamCs)>>`N7%o zY@O`uVSQf~x^52MTJ)eO=*^Yx!)_{7z?7PC>!z4iGVBE7tGTYoIcCt)A_j{jH z2!A`}C-GkiCWNf9x`i|$*?3TOD>{;$t9s%o(I~d(C?A_oqk@Yc!amZHP0{=?tqjN zKgva218VxmgRb8@*^~`wGn*FWr6_Oy9HW!?C%={1C)!p^rtBl{VT&DH-3UVZ1F?pc zv0)*_=(}mUayxTP0CYi`{arM+;L*j8)k6pmMQole4RDuR|3~@2*>19 zW2g&Hg2OtjiM;Koh;CY(<8~y}N%*tLG#zIRoqr55D|4h2%)HrgV!oYyz-a@s zSe-`->DQy}Q%_wUv{#EWhLzA|t1^-(v&bo(Y_B=Q(#TQw;vd>f)j69$5~NaaD(Bne zA8GiRB3!+*?LczuX`d~mc@BPO3z+c!m~S>1Tp$KH0AHZ>08cq=Blt-TU$37{4gG9B zFrOxDZ^sSyqwjH_csrZs6MDuniVq5Y;1gO>Ey1A??sRvzre5!8`eiBWS2^%K*OS?4 zP*yWpZU}w^rI1C5+VIUgI1BvauVx{mX8gyQ&lhnI=`OSO7vMfG01c*sk{k3C$i6ZO z#b>N9%~t)`^svr22<$r)*uU11DR^?qCHrb8C~ZuX-ulk%%yr@W+qBrMNK8I6dCh7M zkV{5goREkKTITup7~T$N5x6VitC{_;zB8CCS6P0`hHvMPp{iVe(nMmV_xr??;5p2>29-gJsbkmJb6I_I4#Q9sR{rk z*hsbbDLY*WNkoyQ+Tb|5h@fA^y5+Zxd^UxPhhm3R0$lrkAk`XPmY`M5g#Ow3+(me? zEWr{OzrI4e)dK4H#sITm7p%NRe`%D34DuVPt!~%M+R8{3L}hy@SL=Mf@lOQNgNaS# zkZ^6FLv2p6=&PvdnP@AybSf>VCrvvCcp+soId!o-ELMsB20q27uUVPI)a);u6k$=X zY2+G%SNiURDW}s^InDt(L@()y8)M60{z?G_S9m6OdoAH>d+o%{pU(K{eeUu_Wram{8AE-OFHN;VDz&}cEpl?iWi z;zOaYvq3zVoNDjD5NF72<(i67*=7BqruoOGYw}PV=aamd&6qaKT5fa@!u3Kspg4E; zCXeItu-be8-P|FUY4!4fx|7b-eIg$lTSv}3!MpL%NosTGqYA-DjCOyPmucIlDlXDc zz!aEp2`k#9KB~_o1}pYMF+%T%(KZ)1Mf{iFatnO;{p6?Iy&$>5`u2f(VPc#MpG67!5)oU%MNB=+uP=jz&$f)$b1)(9mL=MbPHxe3A2n%KdBLmIE4mfO zYYS-3eBBs4^eJucs)QA*Kpq`;5fuxU9zQ;g`u)8RFCoCH2(l_<-EowHrk!Qdgfit^ zJtK)iC0>k(6b1rU3Xu5O{x#bR=)}cC__pKSg_TUlDvupre(aRc`f#{LPG3{boYbU$ z3b5pip3{E43xffwf&(lVp#%hUw~B_S=-Hc&(;B=u+G&c=i_S z^1OG#%w2sw{cPh*hZPmv<7WCe5154n&mhbKIkK_kwj!8eMnX7nG{I93uI;F z*jp>AGitU92Vc6w_oteVKfd0qU>!9wecpoFPrpU%Z3b&8ZixM=pb=Ji`Ecuz;6iz$ z>Z)XOuy!~8MgGAg$FJ-9`RAWrJesfFYpUy;*tA2Zdr7d6c#;yL7xNUB?@<7Ir84z+ zrp`eez=n%BpBm2osMzVlM`g_%R61)sAN<$>N1$wRh#e`wL#ZOgz>%Ja0+-8%+8yV+ z7?NJw9FmFU7Jmc>eSI$u>zg0_(s{LMVhIzzWqko>K($r!d(5QyjSM_z+MFFTFLaE- z=rq(>j+93C2nTdZL_3dcC19sVTS6x5wPJi`7g}%O_?${mk3vo)O^6T;-4u1eaA*_Zc)MX7nPU);pSTUy`CT5d6Wnk9Z zXFwI0eaA4TeJ8-T4d)d}bo7%>P+SD0&0Bzr$LkX+$uSxh&+)-ujM5d_q%xnvW9vEE zy{Pd*JuqlVcDrU^R^>EJZ%c*URFVUlZs^NfUWL7c`d&2cBKS(9ZIHlaW>vbNG+GK2 z#QWnJOc*BX%11F~?l!1}$ndvqs{_MF-2>~{B_S1lYFik4r*Fr~%rRCuWe@0^Lkhf| z4)f=T{4upm*1KVJ{w3qNnx?g+KljvU0L>(Yo-i&?69~_aNv5dUJa?A58Ig0{PjCu6 z*lU5KG53{0B9uh+pER)k`$qeg7U6$Mn?SZ#TiT($o2t{BOSh8?LJu8iCF!+0o0;i- z2a`!!#cL*#Rsqmo>Em)_cHx>v!^CFOq;X5DW47oU$VK+9ugtbfS?X~{037{5Fxg{t z_eCwxNW(NY%h!3qZCj%xOAM|*?RSyoH#u_jcus=i)yGDog8X> zi(19LRCSH8J(q%LFi*aN(0N&I&*u7%)P{O(A+ITbnftgah4lX$CGKd@)k#* zKy&1Q+q)e@BPOZ7s8Nm%{dHI+3M+Q{$jd9z*^sI9K@@UzEdGH>3w)$r)4JS4f}%>r)=8)FjBzmEn$5HB;EWOA6P5dzwbFzsR#jmQQHERqL(hn43oNX7 z*#K3os<5yv>CeWtkb(uzG)EI~&kVn=gEpOI+5dejKO5YY@^d!mns`iI_pL)uH(xxGGD=V{-44n2G( zLNs5!AuKf>BUvzfR%gVplub(W(4}=jAUkcP@3Jvqqh1n~p3^*mM%)%blqh6;3Fr=& zrYk>M+G)&Q*Io?i!_M?4F@XaVPYr&$&|DHXc4OoYL4=)jS z)L%RtFiuUQ?1FOh)*OgA9s1o}eS4d}&cWixwJvY_e-_BRxrT4`Ca%75P}$LxpwE6m z+ns^j)xUHJ3WybS(hrx0d-x$weN&}e^;_M_2?*3BXi;wNk7%}o$v}0D7iw==gF^Sk z$Lo!{!?b|bUwB~g7|Twp+tA71$mU;Xw8)bZd)I%rcxQ$DX?Pj!^Q^y-bx-Rdp4a3Z z9YK-x07Ib|-#!Ip*+Z5k!k)TNp&~U*H|ui}usZ|w$hrO@4q~<1IIiy9;$bed{b*fz!nywnJk^ixsW#Y+}I`xa%{yD|O>fsU+!3O4_~vwuAo(6CTi@{cgW=)dZ%P%-y> zThgyc8vYRC^mgI4FmuXRhJDMUV5Jcv7H}7VJuk|_Foro(#6;QEt7g+7bCiJ0ho{;| zd=l9V+Fq-v+%OEB_vlJr^>AZcZlt;WwUnjN8(`06E5z)KLKaBig4KnEEw3C->otWno*IC#dy>m^ZhN3%GKl}X@N*^NE*3)`+ z8e#2%9YiiY4cnW;Z!g?ex9`d`^j2>u+ZT81@6A1yJ-_`Ec>X1zJz5cy-IU~tLOcHV zQTnHu`MM*x^j(^K1G~sW1gOVRXm@ie5+URH++Qp&N@2O13^2Jx!vtp0X1Cvh+S_kY z2=_Q#Il1$!hgJrel(ar<{vKRCaY8>b{1P)eU5d)6^(Z${&8*zh&5#hSTgJ`!+GQ%P z83`<>S@n{{ym7wF27suNlt;nK$8mJ9xXt=n8Ji00tN9t9blDDj3J|cjf&MbDzt8bF z0%U-RXZ0UK-RpGb$%~(@d)UD<>@@PhNw!Ph{yC#T%^>akz^O&g=ED$d^C*qrvSggd zYFxusT|}&)wLB@Z{?vPQrDDpvzRm>yA~Bu0P$Z}hOj;SOpS8rm+q0j4+VN#5u-T`M zHb=F3K|ssGaLS2jL4@Kss80S3NbAA;oyqv-Uv#QHB{!tBw)FoL!vhpOkNetR1}9n8 zZP{Jr4da{MA{h>^hWWY!1y)Nrm)}C@%6$Mtt@gn8&|RRJ!&mUmQ>j*t>loA(DtOcE z@;v;i1QXzAKJ7H8&Y_W=`DmnPclT)HTa}p4sTg`$XuL-1hYb3SWPLxjasj}~kv3jY zK;U}e3~NGixo|siTSSTD^mZ%TGiqvR+X18%N71y~+O4>LYy*T_r&3*4X9@fx2viPkgA|d80pxmBNT1}kZdh=P@Wo5wjJZSG z%dF2K<+PaP8_Twc7>dZ0eJOQ;gs-@_o>#of7Zwh|q zx<*v))Plyko%(!=mdbNSfHL+AuE zoAj$`oNu1YZ&fWnQSn+w>~Uxt^_y^YxtD@J07t0U!%B{lcwm9JTZk9f z#lVrvZe9WTY<2_vZa7Z|)#k?zAU^F-rFG$r7smGsoF#n?3wXcgWDBoKKHjxIZ&@490_shOqD`K>chq~uJh-l+mVf|gl>ewf(xg&#?`B`U9G-nRx z^!))Z<+VObe#7Br2LEB|MQ1Z@MO=%a;OX&k#Lh}akh(+$=V%jb=lxbiY@q#bkm9FS9GtZ;0J!MH+QP{ASyRo3<4 zuZJ`7Ol0|25#P+y9!^j=YMYqZ1a?rmDmYIYkjZgTerF-?jyPtZtfd z9vS(iiXA7Qx&S~K>2kt|zX(z$B#F{>e8w*=1HciLjR4tsu+`t17?VMA2i&!W^5KBF zU9~ER73qtYL`lsq^pYGPFQz&!{LKFrW~9SpP~Z=aZs6!(QC3WW7x1=)>sFM6?B8nn zqQR?>m*4fKn=vmHOjA=JzYOQ%VASq)fHflTPvke++;(W=|HcwQdUH9b#^Qf`=}oah z=;_j0Fa7G@0$tCF!id~DHT1IKycS&TN zXgj8ctlLw#roa(+K|%H8l)P762M6w=Zm;88Je>7g%a?NNkoF(fewnP#%Y?tVMuBX1 z461MqXY{g1H$smTNTE(nla*zjG<5?I1v}6zO`R)5R6<6t0ch&ZBaCzbw3utUPT1L1AvAJTga!- zu>Gbi{wAEh2Bt%$#Z?k?JhjGsX$Y|rE43UsiDwEQ9u;_apWJA@#E$a;q#Qv@?U?gm ztw)xMLQP^F7hK`a$b%ARCj7a6EV|wNU0~AsRj||B$QjbfKI}_s$jNr4X|$6aHC*Qq zvMI~OHKcBY&7L&qC@-DMR*I~kI_Rd|Nxo9-YafHev@SKtCJ{HtvSD9pzl~GL?H!&Z>Ewr3Yq2 z3F4%@T%63XzNs7%#BiocDd|g+3#InK?oQ0FeUf<(PORM*^n3FJiT;x=ei=eMGHb!U z5;t8o>^t4(jU8B28zRe(^MOXe%tF`y8%G;^!fMV^XKsMzbYgC z9FzWOn)BySLg0V)P7S>;DcM{f`st-xSjS z@FpX2U;URr^v?)=zWzr5@aOpd=yCgBKCAOL&CmBDj;If^eL~jyj|e*dt0m<>FEIb* z(SKic{+#CGtKS=lKZpPPtLpC0F=Ohk3!ar)k)kB~cxT7IruV;mqyIGL|Kpqc4bgvS zq<{X8uf=b*`2Xgw(*90o|8ITy&xiis%m2+IK;Y@av-b*MHFxB+t<9tSgd}?e{(qmo ze=kFS9!UxKau(}<7@dD^-uS)p|9Ow`|KxG||ICp8@yku5h?6CXjRz}|n9GrbjXTi) wiWB!=!sGt)o8Ni75WjVke@^p%X(X { // Fetch new access token with in storage refresh token const body = new URLSearchParams() - body.append('client_id', process.env.CLIENT_ID || '') + body.append('client_id', apiConfig.clientId) body.append('redirect_uri', apiConfig.redirectUri) body.append('client_secret', clientSecret) body.append('refresh_token', refreshToken) diff --git a/src/pages/api/item.ts b/src/pages/api/item.ts old mode 100644 new mode 100755 diff --git a/src/pages/api/name/[name].ts b/src/pages/api/name/[name].ts old mode 100644 new mode 100755 diff --git a/src/pages/api/raw.ts b/src/pages/api/raw.ts old mode 100644 new mode 100755 diff --git a/src/pages/api/search.ts b/src/pages/api/search.ts old mode 100644 new mode 100755 diff --git a/src/pages/api/thumbnail.ts b/src/pages/api/thumbnail.ts old mode 100644 new mode 100755 diff --git a/src/pages/index.tsx b/src/pages/index.tsx old mode 100644 new mode 100755 diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx old mode 100644 new mode 100755 index 783b28fe26..5a6496f391 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -11,41 +11,7 @@ import Footer from '../../components/Footer' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' import { getAccessToken } from '../api' // 导入getAccessToken函数 -function obfuscateSensitiveData(data) { - if (data.length <= 12) { - return data; - } - const start = data.substring(0, 6); - const end = data.substring(data.length - 6); - return `${start}******${end}`; -} - -export async function getServerSideProps({ locale }) { - const clientId = process.env.CLIENT_ID || ''; - const clientSecret = process.env.CLIENT_SECRET || ''; - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - - // 如果访问令牌存在,重定向到主页 - if (accessToken) { - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - - // 如果访问令牌不存在,正常渲染页面 - return { - props: { - ...(await serverSideTranslations(locale, ['common'])), - clientId, - clientSecret, - }, - } -} - -export default function OAuthStep1({ clientId, clientSecret }) { +export default function OAuthStep1() { const router = useRouter() const { t } = useTranslation() @@ -108,7 +74,7 @@ export default function OAuthStep1({ clientId, clientSecret }) { CLIENT_ID - {obfuscateSensitiveData(clientId)} + {apiConfig.clientId} @@ -116,7 +82,7 @@ export default function OAuthStep1({ clientId, clientSecret }) { CLIENT_SECRET* - {obfuscateSensitiveData(clientSecret)} + {apiConfig.obfuscatedClientSecret} @@ -181,3 +147,23 @@ export default function OAuthStep1({ clientId, clientSecret }) {
) } + +export async function getServerSideProps({ locale }) { + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + // 如果访问令牌不存在,正常渲染页面 + return { + props: { + ...(await serverSideTranslations(locale, ['common'])), + }, + } +} + diff --git a/src/pages/onedrive-vercel-index-oauth/step-2.tsx b/src/pages/onedrive-vercel-index-oauth/step-2.tsx old mode 100644 new mode 100755 index 6782223e8c..dd4fd70e00 --- a/src/pages/onedrive-vercel-index-oauth/step-2.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-2.tsx @@ -1,7 +1,7 @@ import Head from 'next/head' import Image from 'next/image' import { useRouter } from 'next/router' -import { useState, useEffect } from 'react' +import { useState } from 'react' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' import { useTranslation, Trans } from 'next-i18next' import { serverSideTranslations } from 'next-i18next/serverSideTranslations' @@ -13,25 +13,6 @@ import { LoadingIcon } from '../../components/Loading' import { extractAuthCodeFromRedirected, generateAuthorisationUrl } from '../../utils/oAuthHandler' import { getAccessToken } from '../api' // 导入getAccessToken函数 -export async function getServerSideProps({ locale }) { - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - // 如果访问令牌存在,重定向到主页 - if (accessToken) { - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - // 如果访问令牌不存在,正常渲染页面 - return { - props: { - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} - export default function OAuthStep2() { const router = useRouter() @@ -41,13 +22,7 @@ export default function OAuthStep2() { const { t } = useTranslation() - // const oAuthUrl = generateAuthorisationUrl() - - const [oAuthUrl, setOAuthUrl] = useState(null) - - useEffect(() => { - generateAuthorisationUrl().then(url => setOAuthUrl(url)) - }, []) + const oAuthUrl = generateAuthorisationUrl() return (
@@ -85,9 +60,7 @@ export default function OAuthStep2() {
{ - if (oAuthUrl) { - window.open(oAuthUrl) - } + window.open(oAuthUrl) }} >
@@ -168,3 +141,22 @@ export default function OAuthStep2() {
) } + +export async function getServerSideProps({ locale }) { + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + // 如果访问令牌不存在,正常渲染页面 + return { + props: { + ...(await serverSideTranslations(locale, ['common'])), + }, + } +} diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx old mode 100644 new mode 100755 index 16e723f426..19ccc2eaa3 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -14,62 +14,7 @@ import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer } from ' import { LoadingIcon } from '../../components/Loading' import { getAccessToken } from '../api' // 导入getAccessToken函数 -export async function getServerSideProps({ query, locale }) { - const { authCode } = query - const userPrincipalName = process.env.USER_PRINCIPLE_NAME || ''; - - // 检查是否已经通过OAuth认证 - //const accessToken = await getAccessToken(); - //if (accessToken) { - //如果已经通过OAuth认证,重定向到主页 - //return { - //redirect: { - //destination: '/', - //permanent: false, - //}, - //} - //} - - // 如果没有通过OAuth认证,继续执行OAuth认证的流程 - if (!authCode) { - return { - props: { - error: 'No auth code present', - description: 'Where is the auth code? Did you follow step 2 you silly donut?', - ...(await serverSideTranslations(locale, ['common'])), - }, - } - } - - const response = await requestTokenWithAuthCode(authCode) - - // If error response, return invalid - if ('error' in response) { - return { - props: { - error: response.error, - description: response.errorDescription, - errorUri: response.errorUri, - ...(await serverSideTranslations(locale, ['common'])), - }, - } - } - - const { expiryTime, accessToken, refreshToken } = response - - return { - props: { - userPrincipalName, - error: null, - expiryTime, - accessToken, - refreshToken, - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} - -export default function OAuthStep3({ userPrincipalName, accessToken, expiryTime, refreshToken, error, description, errorUri }) { +export default function OAuthStep3({ accessToken, expiryTime, refreshToken, error, description, errorUri }) { const router = useRouter() const [expiryTimeLeft, setExpiryTimeLeft] = useState(expiryTime) @@ -111,7 +56,7 @@ export default function OAuthStep3({ userPrincipalName, accessToken, expiryTime, ) return } - if (data.userPrincipalName !== userPrincipalName) { + if (data.userPrincipalName !== siteConfig.userPrincipalName) { setButtonError(true) setButtonContent(
@@ -278,3 +223,57 @@ export default function OAuthStep3({ userPrincipalName, accessToken, expiryTime,
) } + +export async function getServerSideProps({ query, locale }) { + const { authCode } = query + + // 检查是否已经通过OAuth认证 + const accessToken = await getAccessToken(); + if (accessToken) { + 如果已经通过OAuth认证,重定向到主页 + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + // 如果没有通过OAuth认证,继续执行OAuth认证的流程 + + // Return if no auth code is present + if (!authCode) { + return { + props: { + error: 'No auth code present', + description: 'Where is the auth code? Did you follow step 2 you silly donut?', + ...(await serverSideTranslations(locale, ['common'])), + }, + } + } + + const response = await requestTokenWithAuthCode(authCode) + + // If error response, return invalid + if ('error' in response) { + return { + props: { + error: response.error, + description: response.errorDescription, + errorUri: response.errorUri, + ...(await serverSideTranslations(locale, ['common'])), + }, + } + } + + const { expiryTime, accessToken, refreshToken } = response + + return { + props: { + error: null, + expiryTime, + accessToken, + refreshToken, + ...(await serverSideTranslations(locale, ['common'])), + }, + } +} diff --git a/src/styles/globals.css b/src/styles/globals.css old mode 100644 new mode 100755 diff --git a/src/styles/markdown-github.css b/src/styles/markdown-github.css old mode 100644 new mode 100755 diff --git a/src/types/index.d.ts b/src/types/index.d.ts old mode 100644 new mode 100755 diff --git a/src/utils/fetchOnMount.ts b/src/utils/fetchOnMount.ts old mode 100644 new mode 100755 diff --git a/src/utils/fetchWithSWR.ts b/src/utils/fetchWithSWR.ts old mode 100644 new mode 100755 diff --git a/src/utils/fileDetails.ts b/src/utils/fileDetails.ts old mode 100644 new mode 100755 diff --git a/src/utils/getBaseUrl.ts b/src/utils/getBaseUrl.ts old mode 100644 new mode 100755 diff --git a/src/utils/getFileIcon.ts b/src/utils/getFileIcon.ts old mode 100644 new mode 100755 diff --git a/src/utils/getPreviewType.ts b/src/utils/getPreviewType.ts old mode 100644 new mode 100755 diff --git a/src/utils/getReadablePath.ts b/src/utils/getReadablePath.ts old mode 100644 new mode 100755 diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts old mode 100644 new mode 100755 index 8197141db5..4fde3d5d3d --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -3,11 +3,6 @@ import CryptoJS from 'crypto-js' import apiConfig from '../../config/api.config' -async function getConfig() { - const res = await axios.get('/api/config') - return res.data -} - // Just a disguise to obfuscate required tokens (including but not limited to client secret, // access tokens, and refresh tokens), used along with the following two functions const AES_SECRET_KEY = 'onedrive-vercel-index' @@ -23,10 +18,8 @@ export function revealObfuscatedToken(obfuscated: string): string { } // Generate the Microsoft OAuth 2.0 authorization URL, used for requesting the authorisation code -export async function generateAuthorisationUrl(): Promise { - const config = await getConfig() - const clientId = config.clientId - const { redirectUri, authApi, scope } = apiConfig +export function generateAuthorisationUrl(): string { + const { clientId, redirectUri, authApi, scope } = apiConfig const authUrl = authApi.replace('/token', '/authorize') // Construct URL parameters for OAuth2 @@ -62,10 +55,8 @@ export async function requestTokenWithAuthCode( | { expiryTime: string; accessToken: string; refreshToken: string } | { error: string; errorDescription: string; errorUri: string } > { - const config = await getConfig() - const clientId = config.clientId - const clientSecret = revealObfuscatedToken(config.clientSecret) - const { redirectUri, authApi } = apiConfig + const { clientId, redirectUri, authApi } = apiConfig + const clientSecret = revealObfuscatedToken(apiConfig.obfuscatedClientSecret) // Construct URL parameters for OAuth2 const params = new URLSearchParams() diff --git a/src/utils/odAuthTokenStore.ts b/src/utils/odAuthTokenStore.ts old mode 100644 new mode 100755 diff --git a/src/utils/protectedRouteHandler.ts b/src/utils/protectedRouteHandler.ts old mode 100644 new mode 100755 index afa99b625a..6e2bbdb333 --- a/src/utils/protectedRouteHandler.ts +++ b/src/utils/protectedRouteHandler.ts @@ -55,4 +55,4 @@ export function matchProtectedRoute(route: string): string { } } return authTokenPath -} \ No newline at end of file +} diff --git a/src/utils/useDeviceOS.ts b/src/utils/useDeviceOS.ts old mode 100644 new mode 100755 diff --git a/src/utils/useLocalStorage.ts b/src/utils/useLocalStorage.ts old mode 100644 new mode 100755 diff --git a/tailwind.config.js b/tailwind.config.js old mode 100644 new mode 100755 diff --git a/tsconfig.json b/tsconfig.json old mode 100644 new mode 100755 From 71a8a831da572b1c0a217a6d6d21b586f834ed3e Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Fri, 7 Jul 2023 15:44:19 +0800 Subject: [PATCH 35/88] bug fixed --- README.zh-CN.md | 2 +- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index f31d1df013..2686926ff6 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,7 +36,7 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPLE_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 19ccc2eaa3..d8bee38b8c 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -230,7 +230,7 @@ export async function getServerSideProps({ query, locale }) { // 检查是否已经通过OAuth认证 const accessToken = await getAccessToken(); if (accessToken) { - 如果已经通过OAuth认证,重定向到主页 + // 如果已经通过OAuth认证,重定向到主页 return { redirect: { destination: '/', From 50cad7c831693eb18a6544c4b51bb417cd55d576 Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Fri, 7 Jul 2023 15:53:05 +0800 Subject: [PATCH 36/88] Update step-3.tsx --- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index d8bee38b8c..c178e4390c 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -228,8 +228,8 @@ export async function getServerSideProps({ query, locale }) { const { authCode } = query // 检查是否已经通过OAuth认证 - const accessToken = await getAccessToken(); - if (accessToken) { + const existingAccessToken = await getAccessToken(); + if (existingAccessToken) { // 如果已经通过OAuth认证,重定向到主页 return { redirect: { From 0cd256d0e13b07f264dba2c5814a4c647b7a8903 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Fri, 7 Jul 2023 16:15:31 +0800 Subject: [PATCH 37/88] Update site.config.js --- config/site.config.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/site.config.js b/config/site.config.js index 6236c04843..369daa07b3 100755 --- a/config/site.config.js +++ b/config/site.config.js @@ -24,7 +24,7 @@ module.exports = { // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. - maxItems: 100, + maxItems: 200, // [OPTIONAL] We use Google Fonts natively for font customisations. // You can check and generate the required links and names at https://fonts.google.com. From b51ea21249c72b7b5dc77c77da85c327fb329e16 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Fri, 7 Jul 2023 23:23:37 +0800 Subject: [PATCH 38/88] Update site.config.js --- config/site.config.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/site.config.js b/config/site.config.js index 369daa07b3..3e687c4bfa 100755 --- a/config/site.config.js +++ b/config/site.config.js @@ -7,7 +7,7 @@ module.exports = { // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about // your email being exposed in public. - userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPLE_NAME || '', + userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPAL_NAME || '', // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. From a9d99799bef03480ae62e273bd7d5a8aa28ce2e1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 00:18:10 +0800 Subject: [PATCH 39/88] =?UTF-8?q?=E6=9B=B4=E6=96=B0=20README.zh-CN.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.zh-CN.md | 53 ++++++++++++++++++++++++++++--------------------- 1 file changed, 30 insertions(+), 23 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 2686926ff6..593f896b1f 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -12,6 +12,29 @@ ![demo](./public/demo.png) +## 修改说明 + +- 本版本主要把原本需要在`config/`目录下的`api.config.js`和`site.config.js`这两个配置文件中设置的一些参数搬到了Vercel的环境变量中进行设置,如此便无须——先fork原仓库——然后手动修改配置文件——再部署,而是可以直接点击本文档中的一键部署按钮,在部署过程中输入环境变量的值,然后完成部署。 + +- 再就是本版本设定了当完成OAuth认证后,自动关闭OAuth认证通道,以防有心人通过OAuth认证的网址链接就轻易地获取到用户的配置信息。 + +**在环境变量中设置的配置参数** + +| 名称 | 描述 | 原路径 | 说明 | +| --- | +| **必要参数** | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `config/site.config.js` | 例如:尼加拉瓜首富的OneDrive | +| `NEXT_PUBLIC_USER_PRINCIPLE_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | +| `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | +| `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | +| `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | +| --- | +| 可选参数 | +| `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹 | `config/site.config.js` | 格式:`/route1,/route2` 多个路径使用`,`间隔 | +| `NEXT_PUBLIC_EMAIL` | 您的联系用Email | `config/site.config.js` | example@example.com | +| `KV_PREFIX` | 用于键值对存储的前缀 | `config/site.config.js` | 如果您想要使用同一个`Redis`数据库部署多个OneDrive-Index,那么你就可以在部署时设置这个环境变量,例如第一个Index的`KV_PREFIX`可以设置为`index1`,第二个Index的`KV_PREFIX`可以设置为`index2`,那么它们在Vercel部署时就不会有键值冲突了 | + ## 部署方法 ### 前期准备 @@ -20,27 +43,19 @@ 本项目是通过调用OneDrive的API来获取文件列表以及下载链接的,所以设置OneDrive帐户的API权限是必须的,获取方法请参考原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-api-权限)。 - 需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。 + **需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。** -2. **准备好在Vercel部署时填写的五个环境参数:** - -| 名称 | 描述 | 默认 | 备注 | -| --- | --- | --- | --- | -| `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `null` | 例如:尼加拉瓜首富的OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPLE_NAME` | 您的OneDrive帐户 | `null` | **字母大小写必须一致** | -| `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `null` | (格式为`/目录名`),根目录则填写`/` | -| `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `null` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | -| `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `null` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | +2. **准备好在Vercel部署时填写的五个必要环境参数:** ### 部署到Vercel **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIP AL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 -- `REDIS_URL`:如果您和我一样第一次接触这个Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好,它会自动填入项目部署后的环境变量中。 +- `REDIS_URL`:如果您和我一样第一次接触这个Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 - `REDIS_URL`设置成功后,再重新部署一次项目。 @@ -50,19 +65,11 @@ **更多玩法请查阅原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/getting-started)** -## 修改说明 - -- 本版本对比原版主要是把需要先修改`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`,以及修改`config/site.config.js`中的`userPrincipalName`、`title`和`baseDirectory`的步骤提取出来,放在Vercel部署时的环境变量设置中进行。 - -- 留空了`config/site.config.js`中的`mail`(如果想在页面中展示自己的联系方式,可自行修改),以及去除了GitHub图标旁的`GitHub`字样(因为感觉导航栏右边的图标有点多有点挤了)。 - -- 另外就是加入了[Vercel Analytics](https://vercel.com/docs/concepts/analytics)的支持,方便查看分享的页面被访问情况(需要在部署后自行在项目的Analytics选项卡中开启)。 - ## 安全风险和一些问题 -- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。应该避免在以`NEXT_PUBLIC_`开头的环境变量中存储敏感信息,如API密钥或数据库密码。这些信息应该只在服务器端代码中使用,并且应该使用不带`NEXT_PUBLIC_`前缀的环境变量来存储。 +- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 -- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证的第一步时,无法获取到`clientId`和`obfuscatedClientSecret`的值,为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`obfuscatedClientSecret`在没有OneDrive帐户的登录密码时,也不算是太敏感的信息,就暂时这样解决了。 +- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,出现各种各样的问题而无法完成OAuth认证。为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`ClientSecret`在没有OneDrive帐户的登录密码时,也不会有太大问题,并且也设定了当完成OAuth认证后无法再访问OAuth认证页面轻易获取这些敏感信息,就暂时这样解决了。 ## License @@ -74,4 +81,4 @@
Made by spencer woo | Modified by iRedScarf -
+
\ No newline at end of file From 0047ad2d693774ee81b6e1606c7664f0735de522 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 00:27:42 +0800 Subject: [PATCH 40/88] Update README.zh-CN.md --- README.zh-CN.md | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 593f896b1f..939b5506c1 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -20,17 +20,18 @@ **在环境变量中设置的配置参数** +**必要参数** | 名称 | 描述 | 原路径 | 说明 | -| --- | -| **必要参数** | | --- | --- | --- | --- | | `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `config/site.config.js` | 例如:尼加拉瓜首富的OneDrive | | `NEXT_PUBLIC_USER_PRINCIPLE_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | | `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | | `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | | `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | -| --- | -| 可选参数 | + +*可选参数* +| 名称 | 描述 | 原路径 | 说明 | +| --- | --- | --- | --- | | `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹 | `config/site.config.js` | 格式:`/route1,/route2` 多个路径使用`,`间隔 | | `NEXT_PUBLIC_EMAIL` | 您的联系用Email | `config/site.config.js` | example@example.com | | `KV_PREFIX` | 用于键值对存储的前缀 | `config/site.config.js` | 如果您想要使用同一个`Redis`数据库部署多个OneDrive-Index,那么你就可以在部署时设置这个环境变量,例如第一个Index的`KV_PREFIX`可以设置为`index1`,第二个Index的`KV_PREFIX`可以设置为`index2`,那么它们在Vercel部署时就不会有键值冲突了 | @@ -51,7 +52,7 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIP AL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 @@ -81,4 +82,4 @@ \ No newline at end of file +
From 1c63336511a783bba3e661cce2e02dd31b3e0da5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 00:31:22 +0800 Subject: [PATCH 41/88] Update README.zh-CN.md --- README.zh-CN.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 939b5506c1..b89c1b05cb 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -32,9 +32,9 @@ *可选参数* | 名称 | 描述 | 原路径 | 说明 | | --- | --- | --- | --- | -| `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹 | `config/site.config.js` | 格式:`/route1,/route2` 多个路径使用`,`间隔 | -| `NEXT_PUBLIC_EMAIL` | 您的联系用Email | `config/site.config.js` | example@example.com | -| `KV_PREFIX` | 用于键值对存储的前缀 | `config/site.config.js` | 如果您想要使用同一个`Redis`数据库部署多个OneDrive-Index,那么你就可以在部署时设置这个环境变量,例如第一个Index的`KV_PREFIX`可以设置为`index1`,第二个Index的`KV_PREFIX`可以设置为`index2`,那么它们在Vercel部署时就不会有键值冲突了 | +| `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹 | `config/site.config.js` | 格式:`/route1,/route2`, 多个路径使用`,`间隔 | +| `NEXT_PUBLIC_EMAIL` | 您的联系用Email | `config/site.config.js` | example@example.com | +| `KV_PREFIX` | 用于键值对存储的前缀 | `config/site.config.js` | 如果您想要使用同一个`Redis`数据库部署多个OneDrive-Index,那么您就可以在部署时设置这个环境变量,例如第一个Index的`KV_PREFIX`可以设置为`index1`,第二个Index的`KV_PREFIX`可以设置为`index2`,那么它们在Vercel部署时就不会有键值冲突了 | ## 部署方法 From 7d0147b5a045124cc47d3ccd2f9a299e5ca69211 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 00:36:13 +0800 Subject: [PATCH 42/88] Update README.zh-CN.md --- README.zh-CN.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index b89c1b05cb..251e0d7f1a 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -34,7 +34,7 @@ | --- | --- | --- | --- | | `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹 | `config/site.config.js` | 格式:`/route1,/route2`, 多个路径使用`,`间隔 | | `NEXT_PUBLIC_EMAIL` | 您的联系用Email | `config/site.config.js` | example@example.com | -| `KV_PREFIX` | 用于键值对存储的前缀 | `config/site.config.js` | 如果您想要使用同一个`Redis`数据库部署多个OneDrive-Index,那么您就可以在部署时设置这个环境变量,例如第一个Index的`KV_PREFIX`可以设置为`index1`,第二个Index的`KV_PREFIX`可以设置为`index2`,那么它们在Vercel部署时就不会有键值冲突了 | +| `KV_PREFIX` | 用于键值对存储的前缀 | `config/site.config.js` | Upstash只提供一个免费的`Redis`数据库,如果想要部署多个OneDrive-Index,就可为不同的Index设置不同的`KV_PREFIX`值,那么它们在Vercel部署时就不会有键值冲突了 | ## 部署方法 From bab02d70923c82aa34a02c0667ced4a3b9c7efb3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 00:39:43 +0800 Subject: [PATCH 43/88] Update README.zh-CN.md --- README.zh-CN.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 251e0d7f1a..062f3b526b 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -32,9 +32,9 @@ *可选参数* | 名称 | 描述 | 原路径 | 说明 | | --- | --- | --- | --- | -| `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹 | `config/site.config.js` | 格式:`/route1,/route2`, 多个路径使用`,`间隔 | -| `NEXT_PUBLIC_EMAIL` | 您的联系用Email | `config/site.config.js` | example@example.com | -| `KV_PREFIX` | 用于键值对存储的前缀 | `config/site.config.js` | Upstash只提供一个免费的`Redis`数据库,如果想要部署多个OneDrive-Index,就可为不同的Index设置不同的`KV_PREFIX`值,那么它们在Vercel部署时就不会有键值冲突了 | +| `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹路径 | `config/site.config.js` | 格式:`/route1,/route2`, 多个路径使用`,`间隔 | +| `NEXT_PUBLIC_EMAIL` | 显示在右上角的联系Email | `config/site.config.js` | `example@example.com` | +| `KV_PREFIX` | 用于KV存储(键值对存储)的前缀 | `config/site.config.js` | Upstash只提供一个免费的`Redis`数据库,如果想要部署多个OneDrive-Index,就可为不同的Index设置不同的`KV_PREFIX`值,那么就不会有键值冲突了 | ## 部署方法 From 2d545f348a5abd1fb419e3b1ab78dc26541700ca Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 00:50:54 +0800 Subject: [PATCH 44/88] Update README.zh-CN.md --- README.zh-CN.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 062f3b526b..1ac7cb0f89 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -24,7 +24,7 @@ | 名称 | 描述 | 原路径 | 说明 | | --- | --- | --- | --- | | `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `config/site.config.js` | 例如:尼加拉瓜首富的OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPLE_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | +| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | | `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | | `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | | `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | From a2c7ce094cb327c58473e460f61f1dc087610094 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 01:31:07 +0800 Subject: [PATCH 45/88] Update README.md --- README.md | 64 +++++++++++++++++++++++++++---------------------------- 1 file changed, 31 insertions(+), 33 deletions(-) diff --git a/README.md b/README.md index f1e24b3508..582cf2c66c 100644 --- a/README.md +++ b/README.md @@ -12,25 +12,41 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ![demo](./public/demo.png) +## Modifications + +- This version mainly moves some parameters that need to be set in the `api.config.js` and `site.config.js` in the `config/` to the environment variables of Vercel for setting. In this way, there is no need to - first fork the original repository - then manually modify the configuration file - and then deploy, but you can directly click the one-click deployment button in this document, enter the value of the environment variable during the deployment process, and then complete the deployment. + +- Another thing is that this version is set to automatically close the OAuth authentication channel after completing OAuth authentication, to prevent people with intentions from easily obtaining user configuration information through the OAuth authentication URL link. + +## Environment Variables + +**Necessary parameters** +| Name | Description | Original Path | Note | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_SITE_TITLE` | Title of the display page | `config/site.config.js` | e.g. Nicaragua's richest man's OneDrive | +| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | Your OneDrive account | `config/site.config.js` | **Case-sensitive** | +| `NEXT_PUBLIC_BASE_DIRECTORY` | The OneDrive directory you want to share | `config/site.config.js` | `/directory name`, root directory is `/` | +| `NEXT_PUBLIC_CLIENT_ID` | The client ID of the app you registered in Microsoft Azure | `config/api.config.js` | The one provided by the original author has expired, it is recommended to register one yourself, the validity period can be set to two years (anyway, you have to set the API permissions of the account, by the way). The acquisition method refers to the [DOCS](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) | +| `NEXT_PUBLIC_CLIENT_SECRET` | The client secret of the app registered in Microsoft Azure | `config/api.config.js` | The acquisition method is the same, especially note that this **needs to encrypt the original secret with AES** (can be done in the [DOCS](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | + +*Optional parameters* +| Name | Description | Original Path | Note | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_PROTECTED_ROUTES` | The path of the folder that needs password access | `config/site.config.js` | Format: `/route1,/route2`, multiple paths are separated by `,` | +| `NEXT_PUBLIC_EMAIL` | Contact Email displayed in the upper right corner | `config/site.config.js` | `example@example.com` | +| `KV_PREFIX` | Prefix for KV storage (key-value pair storage) | `config/site.config.js` | Upstash only provides a free `Redis` database, if you want to deploy multiple OneDrive-Index, you can set different `KV_PREFIX` values for different Index, so there will be no key value conflict | + ## Getting Started ### Preparations 1. **Setting up the API permissions for your OneDrive account:** - This project retrieves the file list and download links by calling OneDrive's API, so setting up the API permissions for your OneDrive account is essential. Please refer to the [Docs](https://ovi.swo.moe/docs/advanced#modify-api-permissions) written by the original author for retrieval methods. + This project retrieves the file list and download links by calling OneDrive's API, so setting up the API permissions for your OneDrive account is essential. Please refer to the [DOCS](https://ovi.swo.moe/docs/advanced#modify-api-permissions). The three API permissions that need to be set up are: `user.read`, `files.read.all`, `offline_access`. -2. **Prepare the five environmental parameters to be filled in during deployment on Vercel:** - -| Name | Description | Default | Note | -| --- | --- | --- | --- | -| `NEXT_PUBLIC_SITE_TITLE` | Title of the displayed page | `null` | e.g., OneDrive of the Richest Man in Nicaragua | -| `NEXT_PUBLIC_USER_PRINCIPLE_NAME` | Your OneDrive account | `null` | **Case-sensitive** | -| `NEXT_PUBLIC_BASE_DIRECTORY` | The OneDrive directory you want to share | `null` | Format is `/directory-name`, for root directory, fill in `/` | -| `NEXT_PUBLIC_CLIENT_ID` | Client ID of the app you registered in Microsoft Azure | `null` | The one provided by the original author has expired, it's recommended to register your own, which can be valid for up to two years. Retrieval methods are the same as in the [Docs](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) written by the original author | -| `NEXT_PUBLIC_CLIENT_SECRET` | Client Secret of the app you registered in Microsoft Azure | `null` | Retrieval methods are the same as above, note that you need to **AES encrypt the original secret** (can be done as described in the [Docs](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | +2. **Prepare the five necessary environmental parameters to be filled in during deployment on Vercel.** ### Deploying to Vercel @@ -40,7 +56,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. -- `REDIS_URL`:If you are encountering Redis database for the first time like me, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash), it will automatically fill in the environment variables after project deployment. +- `REDIS_URL`:If you are encountering Redis database for the first time, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash)(simply create a new database in the `Redis` of Upstash, then create a new integration in `Vercel Integrations`, and associate the just deployed OneDrive-Index project with the Redis database), it will automatically fill in the environment variables after project deployment. - After `REDIS_URL` is successfully set, redeploy the project again. @@ -48,44 +64,26 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ## Documentation -**For more usage methods, please refer to the [Docs](https://ovi.swo.moe/docs/getting-started) written by the original author.** - -## Modifications - -- Compared with the original version, this version mainly extracts the steps of modifying `clientId` and `obfuscatedClientSecret` in `config/api.config.js`, as well as modifying `userPrincipalName`, `title`, and `baseDirectory` in `config/site.config.js`, and now sets them as environmental variables during Vercel deployment. - -- The specific values of `clientId` and `obfuscatedClientSecret` are hidden on the page of the first step of OAuth authentication and only the first 6 and last 6 characters are displayed for verification. - -- The mail field in config/site.config.js has been left blank, you can modify this yourself if you want to display your contact information on the page (the demo of this version shows an Email icon). Additionally, the word GitHub next to the GitHub icon has been removed (because the icons on the right side of the navigation bar felt a bit too crowded). - -- Also added support for [Vercel Analytics](https://vercel.com/docs/concepts/analytics) to conveniently check the access situation of the shared page (needs to be enabled in the Analytics tab of the project after deployment). +**For more usage methods, please refer to the [DOCS](https://ovi.swo.moe/docs/getting-started) written by the original author.** ## Security Risks -- In both this version and the original archived version, the deployer's OneDrive account `USER_PRINCIPLE_NAME` is leaked in the source code of the webpage. +- In both this version and the original archived version, the deployer's OneDrive account `USER_PRINCIPAL_NAME` is leaked in the source code of the webpage. - In the original archived version, the deployer's `clientId` can be seen in the link used to obtain the authorization code in OAuth Step-2. The `obfuscatedClientSecret` can be seen in the source code of the OAuth Step-1. > This version checks whether authentication has already been passed when performing the OAuth authentication process. If it has, it redirects to the homepage; otherwise, it proceeds with the OAuth authentication process. It attempts to prevent individuals with malicious intent from obtaining the values of `clientId` and `obfuscatedClientSecret` through the link address of OAuth authentication. -- Due to the design decisions of Next.js, environment variables starting with `NEXT_PUBLIC_` are available not only on the server side but also on the client side (browser). This means that any environment variable starting with `NEXT_PUBLIC_` will be included in the built JavaScript file and sent to the user's browser. Therefore, anyone visiting your website can see the values of these environment variables by looking at the source code of the website or network requests. +- Because of the design decision of Next.js, environment variables starting with `NEXT_PUBLIC_` are not only available on the server side, but also on the client side (browser). This means that any environment variable starting with `NEXT_PUBLIC_` will be included in the built JavaScript file and will be sent to the user's browser. Therefore, anyone visiting your website can view the values of these environment variables by viewing the source code of the website or network requests. -> All environment variables used in this version start with `NEXT_PUBLIC_` (otherwise it cannot function properly)... -> -> Initially, there were attempts to use environment variable key names not starting with `NEXT_PUBLIC_`. However, it was not possible to get the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret` for authentication during OAuth authentication, and even `SITE_TITLE` and `BASE_DIRECTORY` were not set as key values of environment variables. For the sake of smooth deployment, these parameters had to temporarily use environment variable key names starting with `NEXT_PUBLIC_`. +- At the beginning, when setting the `clientId` and `obfuscatedClientSecret` in `config/api.config.js` to the environment variables, I have tried to use environment variable key names that do not start with `NEXT_PUBLIC_`, but there will be various problems during OAuth authentication and OAuth authentication cannot be completed. In order to deploy smoothly, I had to use environment variable key names starting with `NEXT_PUBLIC_`. Considering that `clientId` and `ClientSecret` will not have much problem without the login password of the OneDrive account, and it is also set that the OAuth authentication page cannot be accessed easily to obtain these sensitive information after completing OAuth authentication, it is temporarily solved in this way. ## Todo List -- Transition the `protectedRoutes` settings from `config/site.config.js` to environment variables. - - Put the password in the environment variables instead of the `.password` file. - Deepen the study of the original version of the code and strive to implement the function with environment variable key names that do not start with `NEXT_PUBLIC_`, to improve security. -- Close the OAuth authentication pathway after completing the OAuth authentication, striving not to leak the values of `USER_PRINCIPLE_NAME`, `clientId`, and `obfuscatedClientSecret`. - -> It has been preliminarily achieved, but further verification is needed to determine whether the related security risks have been removed. - - Redesign the LOGO. The contrast of the original LOGO is too low, and it is not consistent enough with the style of other icons and fonts on the page. ## License From 3162da4f1365a8be2e513fa021f32cd00f14e0fc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 01:43:25 +0800 Subject: [PATCH 46/88] Update README.zh-CN.md --- README.zh-CN.md | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index 1ac7cb0f89..c573af5efb 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -66,11 +66,25 @@ **更多玩法请查阅原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/getting-started)** -## 安全风险和一些问题 +## 安全风险 + +- 这个版本和原作者的存档版本中,部署者的OneDrive账户`USER_PRINCIPAL_NAME`都会暴露在网页的源代码中。 + +- 在原始的存档版本中,部署者的`clientId`可以在OAuth认证第二步中获取授权码的链接中看到,`obfuscatedClientSecret`可以在OAuth认证第一步的源代码中看到。 + +> 这个版本在执行OAuth认证过程时会检查是否已经通过了认证。如果已经通过,它会重定向到主页;否则,它才会继续进行OAuth认证过程。如此试图阻止有心人通过OAuth认证的链接地址获取`clientId`和`obfuscatedClientSecret`的值。 - 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 -- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,出现各种各样的问题而无法完成OAuth认证。为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`ClientSecret`在没有OneDrive帐户的登录密码时,也不会有太大问题,并且也设定了当完成OAuth认证后无法再访问OAuth认证页面轻易获取这些敏感信息,就暂时这样解决了。 +- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,出现各种各样的问题而无法完成OAuth认证。为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`ClientSecret`在没有OneDrive帐户的登录密码时,也不会有太大问题,并且也设定了当完成OAuth认证后无法轻易访问OAuth认证页面获取这些敏感信息,就暂时这样解决了。 + +## 待办事项 + +- 将密码放在环境变量中,而不是加密目录下的`.password`文件中。 + +- 深入研究原始版本的代码,努力以非`NEXT_PUBLIC_`开头的环境变量键名实现功能,以提高安全性。 + +- 重新设计LOGO。原始LOGO的对比度太低,与页面上其他图标和字体的风格不够一致。 ## License From 65eb79043f7aaecdc24be40ec3c70e4e77b60e5b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=88=91=EF=BC=8C=E7=BA=A2=E9=A2=86=E5=B7=BE=EF=BC=81?= <37926645+iRedScarf@users.noreply.github.com> Date: Sat, 8 Jul 2023 01:50:17 +0800 Subject: [PATCH 47/88] Update site.config.js --- config/site.config.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/site.config.js b/config/site.config.js index 3e687c4bfa..62e4bc3ef4 100755 --- a/config/site.config.js +++ b/config/site.config.js @@ -45,7 +45,7 @@ module.exports = { protectedRoutes: process.env.NEXT_PUBLIC_PROTECTED_ROUTES ? process.env.NEXT_PUBLIC_PROTECTED_ROUTES.split(',') : [], // [OPTIONAL] Use "" here if you want to remove this email address from the nav bar. - email: 'mailto:iredscarf@freeloop.one', + email: process.env.NEXT_PUBLIC_EMAIL ? `mailto:${process.env.NEXT_PUBLIC_EMAIL}` : '', // [OPTIONAL] This is an array of names and links for setting your social information and links. // In the latest update, all brand icons inside font awesome is supported and the icon to render is based on the name From d939cf0e62c50408092caf7c3510fea319f1edbd Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Sat, 8 Jul 2023 02:23:51 +0800 Subject: [PATCH 48/88] Update README.zh-CN.md --- README.zh-CN.md | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index c573af5efb..c7aa06552a 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -18,7 +18,7 @@ - 再就是本版本设定了当完成OAuth认证后,自动关闭OAuth认证通道,以防有心人通过OAuth认证的网址链接就轻易地获取到用户的配置信息。 -**在环境变量中设置的配置参数** +## 环境变量 **必要参数** | 名称 | 描述 | 原路径 | 说明 | @@ -56,7 +56,7 @@ - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 -- `REDIS_URL`:如果您和我一样第一次接触这个Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 +- `REDIS_URL`:如果您是第一次接触Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 - `REDIS_URL`设置成功后,再重新部署一次项目。 @@ -68,11 +68,11 @@ ## 安全风险 -- 这个版本和原作者的存档版本中,部署者的OneDrive账户`USER_PRINCIPAL_NAME`都会暴露在网页的源代码中。 +- 本版本和原作者的存档版本中,部署者的OneDrive账户`USER_PRINCIPAL_NAME`都会暴露在网页的源代码中。 -- 在原始的存档版本中,部署者的`clientId`可以在OAuth认证第二步中获取授权码的链接中看到,`obfuscatedClientSecret`可以在OAuth认证第一步的源代码中看到。 +- 在原作者的存档版本中,部署者的`clientId`可以在OAuth认证第二步中获取授权码的链接中看到,`obfuscatedClientSecret`可以在OAuth认证第一步的源代码中看到。 -> 这个版本在执行OAuth认证过程时会检查是否已经通过了认证。如果已经通过,它会重定向到主页;否则,它才会继续进行OAuth认证过程。如此试图阻止有心人通过OAuth认证的链接地址获取`clientId`和`obfuscatedClientSecret`的值。 +> 本版本在执行OAuth认证过程时会检查是否已经通过了认证,若已通过认证则会重定向到主页,否则才会继续进行OAuth认证过程。如此试图阻止有心人通过OAuth认证的链接地址获取`clientId`和`obfuscatedClientSecret`的值。 - 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 From 2684e8536cedc612c6fafaa6669f4eb10e0a2f8c Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Sat, 8 Jul 2023 09:02:59 +0800 Subject: [PATCH 49/88] Update README.md --- README.md | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 582cf2c66c..a1b79632c5 100644 --- a/README.md +++ b/README.md @@ -29,7 +29,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h | `NEXT_PUBLIC_CLIENT_ID` | The client ID of the app you registered in Microsoft Azure | `config/api.config.js` | The one provided by the original author has expired, it is recommended to register one yourself, the validity period can be set to two years (anyway, you have to set the API permissions of the account, by the way). The acquisition method refers to the [DOCS](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) | | `NEXT_PUBLIC_CLIENT_SECRET` | The client secret of the app registered in Microsoft Azure | `config/api.config.js` | The acquisition method is the same, especially note that this **needs to encrypt the original secret with AES** (can be done in the [DOCS](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | -*Optional parameters* +***Optional parameters*** | Name | Description | Original Path | Note | | --- | --- | --- | --- | | `NEXT_PUBLIC_PROTECTED_ROUTES` | The path of the folder that needs password access | `config/site.config.js` | Format: `/route1,/route2`, multiple paths are separated by `,` | @@ -40,11 +40,11 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ### Preparations -1. **Setting up the API permissions for your OneDrive account:** +1. **Setting up the API permissions for your OneDrive account.** - This project retrieves the file list and download links by calling OneDrive's API, so setting up the API permissions for your OneDrive account is essential. Please refer to the [DOCS](https://ovi.swo.moe/docs/advanced#modify-api-permissions). +- This project retrieves the file list and download links by calling OneDrive's API, so setting up the API permissions for your OneDrive account is essential. Please refer to the [DOCS](https://ovi.swo.moe/docs/advanced#modify-api-permissions). - The three API permissions that need to be set up are: `user.read`, `files.read.all`, `offline_access`. +- The three API permissions that need to be set up are: `user.read`, `files.read.all`, `offline_access`. 2. **Prepare the five necessary environmental parameters to be filled in during deployment on Vercel.** From 4f19bea470d78d0763da1749c6d5aa183d39f36b Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Sat, 8 Jul 2023 09:06:40 +0800 Subject: [PATCH 50/88] Update README.zh-CN.md --- README.zh-CN.md | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/README.zh-CN.md b/README.zh-CN.md index c7aa06552a..abf9537968 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -29,7 +29,7 @@ | `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | | `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | -*可选参数* +***可选参数*** | 名称 | 描述 | 原路径 | 说明 | | --- | --- | --- | --- | | `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹路径 | `config/site.config.js` | 格式:`/route1,/route2`, 多个路径使用`,`间隔 | @@ -42,9 +42,9 @@ 1. **设置OneDrive帐户的API权限:** - 本项目是通过调用OneDrive的API来获取文件列表以及下载链接的,所以设置OneDrive帐户的API权限是必须的,获取方法请参考原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-api-权限)。 +- 本项目是通过调用OneDrive的API来获取文件列表以及下载链接的,所以设置OneDrive帐户的API权限是必须的,获取方法请参考原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-api-权限)。 - **需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。** +- **需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。** 2. **准备好在Vercel部署时填写的五个必要环境参数:** @@ -56,7 +56,7 @@ - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 -- `REDIS_URL`:如果您是第一次接触Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 +> `REDIS_URL`:如果您是第一次接触Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 - `REDIS_URL`设置成功后,再重新部署一次项目。 From 7cfb9ad0af96208e616068737acf2be26a80538b Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Sat, 8 Jul 2023 09:08:08 +0800 Subject: [PATCH 51/88] Update README.md --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index a1b79632c5..c3b7625d68 100644 --- a/README.md +++ b/README.md @@ -56,7 +56,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. -- `REDIS_URL`:If you are encountering Redis database for the first time, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash)(simply create a new database in the `Redis` of Upstash, then create a new integration in `Vercel Integrations`, and associate the just deployed OneDrive-Index project with the Redis database), it will automatically fill in the environment variables after project deployment. +> `REDIS_URL`:If you are encountering Redis database for the first time, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash)(simply create a new database in the `Redis` of Upstash, then create a new integration in `Vercel Integrations`, and associate the just deployed OneDrive-Index project with the Redis database), it will automatically fill in the environment variables after project deployment. - After `REDIS_URL` is successfully set, redeploy the project again. From adebb663d4cb30c588e2ac4321af0520859bcada Mon Sep 17 00:00:00 2001 From: Rock Star <45270927+therockstarind@users.noreply.github.com> Date: Sun, 9 Jul 2023 16:00:37 +0530 Subject: [PATCH 52/88] feat: fetch title for page rendring --- src/pages/[...path].tsx | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/pages/[...path].tsx b/src/pages/[...path].tsx index 0b8dcca8dd..d710313798 100755 --- a/src/pages/[...path].tsx +++ b/src/pages/[...path].tsx @@ -11,11 +11,12 @@ import SwitchLayout from '../components/SwitchLayout' export default function Folders() { const { query } = useRouter() + const title = (query.path && Array.isArray(query.path) ? query.path[query.path.length - 1] : '') return (
- {siteConfig.title} + {title}
From 9aacdc5691dbeb2db15edbeef393bd43297f3bc5 Mon Sep 17 00:00:00 2001 From: eks <11900611@qq.com> Date: Mon, 10 Jul 2023 10:52:01 +0800 Subject: [PATCH 53/88] update README --- README.md | 38 +++++++++++++++++++------------------- README.zh-CN.md | 38 +++++++++++++++++++------------------- 2 files changed, 38 insertions(+), 38 deletions(-) diff --git a/README.md b/README.md index c3b7625d68..52f92a02eb 100644 --- a/README.md +++ b/README.md @@ -18,24 +18,6 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - Another thing is that this version is set to automatically close the OAuth authentication channel after completing OAuth authentication, to prevent people with intentions from easily obtaining user configuration information through the OAuth authentication URL link. -## Environment Variables - -**Necessary parameters** -| Name | Description | Original Path | Note | -| --- | --- | --- | --- | -| `NEXT_PUBLIC_SITE_TITLE` | Title of the display page | `config/site.config.js` | e.g. Nicaragua's richest man's OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | Your OneDrive account | `config/site.config.js` | **Case-sensitive** | -| `NEXT_PUBLIC_BASE_DIRECTORY` | The OneDrive directory you want to share | `config/site.config.js` | `/directory name`, root directory is `/` | -| `NEXT_PUBLIC_CLIENT_ID` | The client ID of the app you registered in Microsoft Azure | `config/api.config.js` | The one provided by the original author has expired, it is recommended to register one yourself, the validity period can be set to two years (anyway, you have to set the API permissions of the account, by the way). The acquisition method refers to the [DOCS](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) | -| `NEXT_PUBLIC_CLIENT_SECRET` | The client secret of the app registered in Microsoft Azure | `config/api.config.js` | The acquisition method is the same, especially note that this **needs to encrypt the original secret with AES** (can be done in the [DOCS](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | - -***Optional parameters*** -| Name | Description | Original Path | Note | -| --- | --- | --- | --- | -| `NEXT_PUBLIC_PROTECTED_ROUTES` | The path of the folder that needs password access | `config/site.config.js` | Format: `/route1,/route2`, multiple paths are separated by `,` | -| `NEXT_PUBLIC_EMAIL` | Contact Email displayed in the upper right corner | `config/site.config.js` | `example@example.com` | -| `KV_PREFIX` | Prefix for KV storage (key-value pair storage) | `config/site.config.js` | Upstash only provides a free `Redis` database, if you want to deploy multiple OneDrive-Index, you can set different `KV_PREFIX` values for different Index, so there will be no key value conflict | - ## Getting Started ### Preparations @@ -46,7 +28,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - The three API permissions that need to be set up are: `user.read`, `files.read.all`, `offline_access`. -2. **Prepare the five necessary environmental parameters to be filled in during deployment on Vercel.** +2. **Prepare the five [necessary environmental variables](#necessary-variables) to be filled in during deployment on Vercel.** ### Deploying to Vercel @@ -62,6 +44,24 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h **After successful deployment, when you visit your `onedrive-vercel-index` page for the first time, it will guide you to perform OAuth authentication (quite simple). For details, please refer to the [Instructions](https://ovi.swo.moe/zh/docs/getting-started#authentication) written by the original author.** +## Environment Variables + +### Necessary Variables +| Name | Description | Original Path | Note | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_SITE_TITLE` | Title of the display page | `config/site.config.js` | e.g. Nicaragua's richest man's OneDrive | +| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | Your OneDrive account | `config/site.config.js` | **Case-sensitive** | +| `NEXT_PUBLIC_BASE_DIRECTORY` | The OneDrive directory you want to share | `config/site.config.js` | `/directory name`, root directory is `/` | +| `NEXT_PUBLIC_CLIENT_ID` | The client ID of the app you registered in Microsoft Azure | `config/api.config.js` | The one provided by the original author has expired, it is recommended to register one yourself, the validity period can be set to two years (anyway, you have to set the API permissions of the account, by the way). The acquisition method refers to the [DOCS](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) | +| `NEXT_PUBLIC_CLIENT_SECRET` | The client secret of the app registered in Microsoft Azure | `config/api.config.js` | The acquisition method is the same, especially note that this **needs to encrypt the original secret with AES** (can be done in the [DOCS](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | + +### Optional Variables +| Name | Description | Original Path | Note | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_PROTECTED_ROUTES` | The path of the folder that needs password access | `config/site.config.js` | Format: `/route1,/route2`, multiple paths are separated by `,` | +| `NEXT_PUBLIC_EMAIL` | Contact Email displayed in the upper right corner | `config/site.config.js` | `example@example.com` | +| `KV_PREFIX` | Prefix for KV storage (key-value pair storage) | `config/site.config.js` | Upstash only provides a free `Redis` database, if you want to deploy multiple OneDrive-Index, you can set different `KV_PREFIX` values for different Index, so there will be no key value conflict | + ## Documentation **For more usage methods, please refer to the [DOCS](https://ovi.swo.moe/docs/getting-started) written by the original author.** diff --git a/README.zh-CN.md b/README.zh-CN.md index abf9537968..99b96b2d2a 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -18,24 +18,6 @@ - 再就是本版本设定了当完成OAuth认证后,自动关闭OAuth认证通道,以防有心人通过OAuth认证的网址链接就轻易地获取到用户的配置信息。 -## 环境变量 - -**必要参数** -| 名称 | 描述 | 原路径 | 说明 | -| --- | --- | --- | --- | -| `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `config/site.config.js` | 例如:尼加拉瓜首富的OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | -| `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | -| `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | -| `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | - -***可选参数*** -| 名称 | 描述 | 原路径 | 说明 | -| --- | --- | --- | --- | -| `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹路径 | `config/site.config.js` | 格式:`/route1,/route2`, 多个路径使用`,`间隔 | -| `NEXT_PUBLIC_EMAIL` | 显示在右上角的联系Email | `config/site.config.js` | `example@example.com` | -| `KV_PREFIX` | 用于KV存储(键值对存储)的前缀 | `config/site.config.js` | Upstash只提供一个免费的`Redis`数据库,如果想要部署多个OneDrive-Index,就可为不同的Index设置不同的`KV_PREFIX`值,那么就不会有键值冲突了 | - ## 部署方法 ### 前期准备 @@ -46,7 +28,7 @@ - **需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。** -2. **准备好在Vercel部署时填写的五个必要环境参数:** +2. **准备好在Vercel部署时填写的五个[必要环境变量](#必要参数):** ### 部署到Vercel @@ -62,6 +44,24 @@ **部署成功后,当您第一次访问您的`onedrive-vercel-index`页面时,会引导你进行OAuth认证(相当简单),详情请参考原作者编写的[说明文档](https://ovi.swo.moe/zh/docs/getting-started#进行认证)。** +## 环境变量 + +### 必要变量 +| 名称 | 描述 | 原路径 | 说明 | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `config/site.config.js` | 例如:尼加拉瓜首富的OneDrive | +| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | +| `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | +| `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | +| `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | + +### 可选变量 +| 名称 | 描述 | 原路径 | 说明 | +| --- | --- | --- | --- | +| `NEXT_PUBLIC_PROTECTED_ROUTES` | 需要密码访问的文件夹路径 | `config/site.config.js` | 格式:`/route1,/route2`, 多个路径使用`,`间隔 | +| `NEXT_PUBLIC_EMAIL` | 显示在右上角的联系Email | `config/site.config.js` | `example@example.com` | +| `KV_PREFIX` | 用于KV存储(键值对存储)的前缀 | `config/site.config.js` | Upstash只提供一个免费的`Redis`数据库,如果想要部署多个OneDrive-Index,就可为不同的Index设置不同的`KV_PREFIX`值,那么就不会有键值冲突了 | + ## 说明文档 **更多玩法请查阅原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/getting-started)** From 0f53ec1f81fc13640c1f90eaf9f2bcefffef21dc Mon Sep 17 00:00:00 2001 From: eks <37926645+iRedScarf@users.noreply.github.com> Date: Mon, 10 Jul 2023 11:07:20 +0800 Subject: [PATCH 54/88] Update README.md --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 52f92a02eb..b69cda90a4 100644 --- a/README.md +++ b/README.md @@ -34,7 +34,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h **Once you're prepared, you can click the button below to deploy:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPLE_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) - After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. From aae806a8cc261c2e6741d68d6e6718e112aa775a Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 16:31:37 +0800 Subject: [PATCH 55/88] Add files via upload --- README.md | 9 ++++++++- README.zh-CN.md | 9 ++++++++- 2 files changed, 16 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index b69cda90a4..4567f767f7 100644 --- a/README.md +++ b/README.md @@ -36,6 +36,13 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +> If you have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +> If you have multiple OneDrive accounts that need to use the same Redis database `+ KV_PREFIX`: +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) +> If you need to deploy multiple OneDrive-Index, and all have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) + - After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. > `REDIS_URL`:If you are encountering Redis database for the first time, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash)(simply create a new database in the `Redis` of Upstash, then create a new integration in `Vercel Integrations`, and associate the just deployed OneDrive-Index project with the Redis database), it will automatically fill in the environment variables after project deployment. @@ -50,7 +57,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h | Name | Description | Original Path | Note | | --- | --- | --- | --- | | `NEXT_PUBLIC_SITE_TITLE` | Title of the display page | `config/site.config.js` | e.g. Nicaragua's richest man's OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | Your OneDrive account | `config/site.config.js` | **Case-sensitive** | +| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | Your OneDrive account | `config/site.config.js` | **Case-sensitive** | | `NEXT_PUBLIC_BASE_DIRECTORY` | The OneDrive directory you want to share | `config/site.config.js` | `/directory name`, root directory is `/` | | `NEXT_PUBLIC_CLIENT_ID` | The client ID of the app you registered in Microsoft Azure | `config/api.config.js` | The one provided by the original author has expired, it is recommended to register one yourself, the validity period can be set to two years (anyway, you have to set the API permissions of the account, by the way). The acquisition method refers to the [DOCS](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) | | `NEXT_PUBLIC_CLIENT_SECRET` | The client secret of the app registered in Microsoft Azure | `config/api.config.js` | The acquisition method is the same, especially note that this **needs to encrypt the original secret with AES** (can be done in the [DOCS](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | diff --git a/README.zh-CN.md b/README.zh-CN.md index 99b96b2d2a..916d874768 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,6 +36,13 @@ [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +> 如果你有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +> 如果你有多个OneDrive帐户需要使用同一个Redis数据库`+ KV_PREFIX`: +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) +> 如果你需要部署多个OneDrive-Index同时都有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) + - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 > `REDIS_URL`:如果您是第一次接触Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 @@ -50,7 +57,7 @@ | 名称 | 描述 | 原路径 | 说明 | | --- | --- | --- | --- | | `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `config/site.config.js` | 例如:尼加拉瓜首富的OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | +| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | | `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | | `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | | `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | From a8cb465e969d7e340a4a7d653427b61c18d09f9d Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 16:34:10 +0800 Subject: [PATCH 56/88] Add files via upload --- README.md | 11 ++++++++--- README.zh-CN.md | 11 ++++++++--- 2 files changed, 16 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 4567f767f7..8c8e812e1e 100644 --- a/README.md +++ b/README.md @@ -36,11 +36,16 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) -> If you have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> - If you have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) -> If you have multiple OneDrive accounts that need to use the same Redis database `+ KV_PREFIX`: +> +> - If you have multiple OneDrive accounts that need to use the same Redis database `+ KV_PREFIX`: +> > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) -> If you need to deploy multiple OneDrive-Index, and all have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> +> - If you need to deploy multiple OneDrive-Index, and all have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) - After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. diff --git a/README.zh-CN.md b/README.zh-CN.md index 916d874768..b71c454433 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,11 +36,16 @@ [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) -> 如果你有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> - 如果你有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) -> 如果你有多个OneDrive帐户需要使用同一个Redis数据库`+ KV_PREFIX`: +> +> - 如果你有多个OneDrive帐户需要使用同一个Redis数据库`+ KV_PREFIX`: +> > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) -> 如果你需要部署多个OneDrive-Index同时都有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> +> - 如果你需要部署多个OneDrive-Index同时都有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 From 38855c358dc63fb262f2ce9c2c4c260f5aa23896 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 16:38:23 +0800 Subject: [PATCH 57/88] Add files via upload --- README.md | 12 ++++++------ README.zh-CN.md | 12 ++++++------ 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/README.md b/README.md index 8c8e812e1e..568894640c 100644 --- a/README.md +++ b/README.md @@ -36,17 +36,17 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) -> - If you have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> - If you have folders that need password protection. > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) with `NEXT_PUBLIC_PROTECTED_ROUTES` > -> - If you have multiple OneDrive accounts that need to use the same Redis database `+ KV_PREFIX`: +> - If you have multiple OneDrive accounts that need to use the same Redis database. > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` > -> - If you need to deploy multiple OneDrive-Index, and all have folders that need password protection `+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> - If you need to deploy multiple OneDrive-Index, and all have folders that need password protection. > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` - After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. diff --git a/README.zh-CN.md b/README.zh-CN.md index b71c454433..fb407f6870 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -36,17 +36,17 @@ [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) -> - 如果你有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES`: +> - 如果你有需要密码保护的目录: > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) with `NEXT_PUBLIC_PROTECTED_ROUTES` > -> - 如果你有多个OneDrive帐户需要使用同一个Redis数据库`+ KV_PREFIX`: +> - 如果你有多个OneDrive帐户需要使用同一个Redis数据库: > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` > -> - 如果你需要部署多个OneDrive-Index同时都有需要密码保护的目录`+ NEXT_PUBLIC_PROTECTED_ROUTES & KV_PREFIX`: +> - 如果你需要部署多个OneDrive-Index同时都有需要密码保护的目录: > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 From 13cda01dc7ddb2ee0c71666471dd0f62c8330c46 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:23:29 +0800 Subject: [PATCH 58/88] Create TEST-BUTTON.md --- TEST-BUTTON.md | 1 + 1 file changed, 1 insertion(+) create mode 100644 TEST-BUTTON.md diff --git a/TEST-BUTTON.md b/TEST-BUTTON.md new file mode 100644 index 0000000000..61cb57080f --- /dev/null +++ b/TEST-BUTTON.md @@ -0,0 +1 @@ +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` From 4898028bfdb4fa41a56b01c7395630f1866d3b6a Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:24:21 +0800 Subject: [PATCH 59/88] Update api.config.js --- config/api.config.js | 11 ----------- 1 file changed, 11 deletions(-) diff --git a/config/api.config.js b/config/api.config.js index 2e2d047d75..0d33cd315d 100755 --- a/config/api.config.js +++ b/config/api.config.js @@ -8,20 +8,9 @@ * In which case you would need to change directLinkRegex. */ -const clientIdEnv = process.env.NEXT_PUBLIC_CLIENT_ID; -if (!clientIdEnv) { - throw new Error('`clientId` is not defined in api.config.js'); -} -const clientSecretEnv = process.env.NEXT_PUBLIC_CLIENT_SECRET; -if (!clientSecretEnv) { - throw new Error('`clientSecret` is not defined in api.config.js'); -} - module.exports = { // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. - clientId: clientIdEnv, - obfuscatedClientSecret: clientSecretEnv, // The redirectUri is the URL that the user will be redirected to after they have authenticated with Microsoft Graph API. // Likewise, you would not need to change redirectUri if you are using your personal Microsoft account with OneDrive International. From 3ce532861b5f7d43647a8e8cdbeeb57169949d4d Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:25:32 +0800 Subject: [PATCH 60/88] Update site.config.js --- config/site.config.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/config/site.config.js b/config/site.config.js index 62e4bc3ef4..c5a173de44 100755 --- a/config/site.config.js +++ b/config/site.config.js @@ -7,7 +7,7 @@ module.exports = { // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about // your email being exposed in public. - userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPAL_NAME || '', + // userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPAL_NAME || '', // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. @@ -20,7 +20,7 @@ module.exports = { title: process.env.NEXT_PUBLIC_SITE_TITLE || 'OneDrive-Vercel-Index', // The folder that you are to share publicly with onedrive-vercel-index. Use '/' if you want to share your root folder. - baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', + // baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. From f320203ee7c9c006229f3f0e311ab0f1fd0bc1d4 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:30:30 +0800 Subject: [PATCH 61/88] Update api.config.js --- config/api.config.js | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/config/api.config.js b/config/api.config.js index 0d33cd315d..18f39f5ebe 100755 --- a/config/api.config.js +++ b/config/api.config.js @@ -8,9 +8,20 @@ * In which case you would need to change directLinkRegex. */ -module.exports = { - // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would - // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. +// const clientIdEnv = process.env.NEXT_PUBLIC_CLIENT_ID; +// if (!clientIdEnv) { +// throw new Error('`clientId` is not defined in api.config.js'); +// } +// const clientSecretEnv = process.env.NEXT_PUBLIC_CLIENT_SECRET; +// if (!clientSecretEnv) { +// throw new Error('`clientSecret` is not defined in api.config.js'); +// } + +module.exports = { module.exports = { + // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would + // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. +// clientId: clientIdEnv, +// obfuscatedClientSecret: clientSecretEnv, // The redirectUri is the URL that the user will be redirected to after they have authenticated with Microsoft Graph API. // Likewise, you would not need to change redirectUri if you are using your personal Microsoft account with OneDrive International. From 76346e17ea6b6cedc0dc6be9c3dc285b28c063f9 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:46:13 +0800 Subject: [PATCH 62/88] Update index.ts --- src/pages/api/index.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/pages/api/index.ts b/src/pages/api/index.ts index ed80e7ac8b..017fd0825b 100755 --- a/src/pages/api/index.ts +++ b/src/pages/api/index.ts @@ -10,8 +10,8 @@ import { compareHashedToken } from '../../utils/protectedRouteHandler' import { getOdAuthTokens, storeOdAuthTokens } from '../../utils/odAuthTokenStore' import { runCorsMiddleware } from './raw' -const basePath = pathPosix.resolve('/', siteConfig.baseDirectory) -const clientSecret = revealObfuscatedToken(apiConfig.obfuscatedClientSecret) +const basePath = pathPosix.resolve('/', process.env.BASE_DIRECTORY || '/') +const clientSecret = revealObfuscatedToken(process.env.CLIENT_SECRET || '') /** * Encode the path of the file relative to the base directory @@ -50,7 +50,7 @@ export async function getAccessToken(): Promise { // Fetch new access token with in storage refresh token const body = new URLSearchParams() - body.append('client_id', apiConfig.clientId) + body.append('client_id', process.env.CLIENT_ID || '') body.append('redirect_uri', apiConfig.redirectUri) body.append('client_secret', clientSecret) body.append('refresh_token', refreshToken) From 4ad08f151a99729e9f1a4b05fb1a4a2ee3b68843 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:47:57 +0800 Subject: [PATCH 63/88] Create config.ts --- src/pages/api/config.ts | 10 ++++++++++ 1 file changed, 10 insertions(+) create mode 100644 src/pages/api/config.ts diff --git a/src/pages/api/config.ts b/src/pages/api/config.ts new file mode 100644 index 0000000000..c38e71be5f --- /dev/null +++ b/src/pages/api/config.ts @@ -0,0 +1,10 @@ +import type { NextApiRequest, NextApiResponse } from 'next' + +export default function handler(req: NextApiRequest, res: NextApiResponse) { + res.status(200).json({ + clientId: process.env.CLIENT_ID || '', + clientSecret: process.env.CLIENT_SECRET || '', + userPrincipalName: process.env.USER_PRINCIPAL_NAME || '', + baseDirectory: process.env.BASE_DIRECTORY || '/' + }) +} From 70d05a55e0c88f097bb1b9b0a298fb06949f94c5 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:49:40 +0800 Subject: [PATCH 64/88] Update TEST-BUTTON.md --- TEST-BUTTON.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/TEST-BUTTON.md b/TEST-BUTTON.md index 61cb57080f..ad5fbc417c 100644 --- a/TEST-BUTTON.md +++ b/TEST-BUTTON.md @@ -1 +1 @@ -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Failoha%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) From b761a0cf9d44cb77c5066d2819863564975a3e2e Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Mon, 10 Jul 2023 23:56:20 +0800 Subject: [PATCH 65/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 4fde3d5d3d..8197141db5 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -3,6 +3,11 @@ import CryptoJS from 'crypto-js' import apiConfig from '../../config/api.config' +async function getConfig() { + const res = await axios.get('/api/config') + return res.data +} + // Just a disguise to obfuscate required tokens (including but not limited to client secret, // access tokens, and refresh tokens), used along with the following two functions const AES_SECRET_KEY = 'onedrive-vercel-index' @@ -18,8 +23,10 @@ export function revealObfuscatedToken(obfuscated: string): string { } // Generate the Microsoft OAuth 2.0 authorization URL, used for requesting the authorisation code -export function generateAuthorisationUrl(): string { - const { clientId, redirectUri, authApi, scope } = apiConfig +export async function generateAuthorisationUrl(): Promise { + const config = await getConfig() + const clientId = config.clientId + const { redirectUri, authApi, scope } = apiConfig const authUrl = authApi.replace('/token', '/authorize') // Construct URL parameters for OAuth2 @@ -55,8 +62,10 @@ export async function requestTokenWithAuthCode( | { expiryTime: string; accessToken: string; refreshToken: string } | { error: string; errorDescription: string; errorUri: string } > { - const { clientId, redirectUri, authApi } = apiConfig - const clientSecret = revealObfuscatedToken(apiConfig.obfuscatedClientSecret) + const config = await getConfig() + const clientId = config.clientId + const clientSecret = revealObfuscatedToken(config.clientSecret) + const { redirectUri, authApi } = apiConfig // Construct URL parameters for OAuth2 const params = new URLSearchParams() From 4697b44306594cb90b1f7d9360cef2d28849c171 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 00:01:26 +0800 Subject: [PATCH 66/88] Update step-1.tsx --- .../onedrive-vercel-index-oauth/step-1.tsx | 51 ++++++++++--------- 1 file changed, 28 insertions(+), 23 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx index 5a6496f391..aedf7a79bb 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -11,7 +11,32 @@ import Footer from '../../components/Footer' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' import { getAccessToken } from '../api' // 导入getAccessToken函数 -export default function OAuthStep1() { +export async function getServerSideProps({ locale }) { + const clientId = process.env.CLIENT_ID || ''; + const clientSecret = process.env.CLIENT_SECRET || ''; + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + + // 如果访问令牌不存在,正常渲染页面 + return { + props: { + ...(await serverSideTranslations(locale, ['common'])), + clientId, + clientSecret, + }, + } +} + +export default function OAuthStep1({ clientId, clientSecret }) { const router = useRouter() const { t } = useTranslation() @@ -74,7 +99,7 @@ export default function OAuthStep1() { CLIENT_ID - {apiConfig.clientId} + {obfuscateSensitiveData(clientId)} @@ -82,7 +107,7 @@ export default function OAuthStep1() { CLIENT_SECRET* - {apiConfig.obfuscatedClientSecret} + {obfuscateSensitiveData(clientSecret)} @@ -147,23 +172,3 @@ export default function OAuthStep1() {
) } - -export async function getServerSideProps({ locale }) { - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - // 如果访问令牌存在,重定向到主页 - if (accessToken) { - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - // 如果访问令牌不存在,正常渲染页面 - return { - props: { - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} - From 10e7819590f809454bfcb7af121a4aa4bc5a9c59 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 00:03:28 +0800 Subject: [PATCH 67/88] Update step-2.tsx --- .../onedrive-vercel-index-oauth/step-2.tsx | 52 +++++++++++-------- 1 file changed, 30 insertions(+), 22 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-2.tsx b/src/pages/onedrive-vercel-index-oauth/step-2.tsx index dd4fd70e00..6782223e8c 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-2.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-2.tsx @@ -1,7 +1,7 @@ import Head from 'next/head' import Image from 'next/image' import { useRouter } from 'next/router' -import { useState } from 'react' +import { useState, useEffect } from 'react' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' import { useTranslation, Trans } from 'next-i18next' import { serverSideTranslations } from 'next-i18next/serverSideTranslations' @@ -13,6 +13,25 @@ import { LoadingIcon } from '../../components/Loading' import { extractAuthCodeFromRedirected, generateAuthorisationUrl } from '../../utils/oAuthHandler' import { getAccessToken } from '../api' // 导入getAccessToken函数 +export async function getServerSideProps({ locale }) { + const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 + // 如果访问令牌存在,重定向到主页 + if (accessToken) { + return { + redirect: { + destination: '/', + permanent: false, + }, + } + } + // 如果访问令牌不存在,正常渲染页面 + return { + props: { + ...(await serverSideTranslations(locale, ['common'])), + }, + } +} + export default function OAuthStep2() { const router = useRouter() @@ -22,7 +41,13 @@ export default function OAuthStep2() { const { t } = useTranslation() - const oAuthUrl = generateAuthorisationUrl() + // const oAuthUrl = generateAuthorisationUrl() + + const [oAuthUrl, setOAuthUrl] = useState(null) + + useEffect(() => { + generateAuthorisationUrl().then(url => setOAuthUrl(url)) + }, []) return (
@@ -60,7 +85,9 @@ export default function OAuthStep2() {
{ - window.open(oAuthUrl) + if (oAuthUrl) { + window.open(oAuthUrl) + } }} >
@@ -141,22 +168,3 @@ export default function OAuthStep2() {
) } - -export async function getServerSideProps({ locale }) { - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - // 如果访问令牌存在,重定向到主页 - if (accessToken) { - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - // 如果访问令牌不存在,正常渲染页面 - return { - props: { - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} From de67889ca847f7b00e1c008c6681a3cacca2d2f6 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 00:09:09 +0800 Subject: [PATCH 68/88] Update step-3.tsx --- .../onedrive-vercel-index-oauth/step-3.tsx | 114 +++++++++--------- 1 file changed, 58 insertions(+), 56 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index c178e4390c..cd5756fb3e 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -14,7 +14,63 @@ import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer } from ' import { LoadingIcon } from '../../components/Loading' import { getAccessToken } from '../api' // 导入getAccessToken函数 -export default function OAuthStep3({ accessToken, expiryTime, refreshToken, error, description, errorUri }) { +export async function getServerSideProps({ query, locale }) { + const { authCode } = query + const userPrincipalName = process.env.USER_PRINCIPAL_NAME || ''; + + // 检查是否已经通过OAuth认证 + const accessToken = await getAccessToken(); + if (accessToken) { + // 如果已经通过OAuth认证,重定向到主页 + return { + edirect: { + estination: '/', + ermanent: false, + }, + } + } + + // 如果没有通过OAuth认证,继续执行OAuth认证的流程 + if (!authCode) { + return { + props: { + error: 'No auth code present', + description: 'Where is the auth code? Did you follow step 2 you silly donut?', + ...(await serverSideTranslations(locale, ['common'])), + userPrincipalName, + }, + } + } + + const response = await requestTokenWithAuthCode(authCode) + + // If error response, return invalid + if ('error' in response) { + return { + props: { + error: response.error, + description: response.errorDescription, + errorUri: response.errorUri, + ...(await serverSideTranslations(locale, ['common'])), + }, + } + } + + const { expiryTime, accessToken, refreshToken } = response + + return { + props: { + userPrincipalName, + error: null, + expiryTime, + accessToken, + refreshToken, + ...(await serverSideTranslations(locale, ['common'])), + }, + } +} + +export default function OAuthStep3({ userPrincipalName, accessToken, expiryTime, refreshToken, error, description, errorUri }) { const router = useRouter() const [expiryTimeLeft, setExpiryTimeLeft] = useState(expiryTime) @@ -56,7 +112,7 @@ export default function OAuthStep3({ accessToken, expiryTime, refreshToken, erro ) return } - if (data.userPrincipalName !== siteConfig.userPrincipalName) { + if (data.userPrincipalName !== userPrincipalName) { setButtonError(true) setButtonContent(
@@ -223,57 +279,3 @@ export default function OAuthStep3({ accessToken, expiryTime, refreshToken, erro
) } - -export async function getServerSideProps({ query, locale }) { - const { authCode } = query - - // 检查是否已经通过OAuth认证 - const existingAccessToken = await getAccessToken(); - if (existingAccessToken) { - // 如果已经通过OAuth认证,重定向到主页 - return { - redirect: { - destination: '/', - permanent: false, - }, - } - } - // 如果没有通过OAuth认证,继续执行OAuth认证的流程 - - // Return if no auth code is present - if (!authCode) { - return { - props: { - error: 'No auth code present', - description: 'Where is the auth code? Did you follow step 2 you silly donut?', - ...(await serverSideTranslations(locale, ['common'])), - }, - } - } - - const response = await requestTokenWithAuthCode(authCode) - - // If error response, return invalid - if ('error' in response) { - return { - props: { - error: response.error, - description: response.errorDescription, - errorUri: response.errorUri, - ...(await serverSideTranslations(locale, ['common'])), - }, - } - } - - const { expiryTime, accessToken, refreshToken } = response - - return { - props: { - error: null, - expiryTime, - accessToken, - refreshToken, - ...(await serverSideTranslations(locale, ['common'])), - }, - } -} From dbddaced0ec539f649ee787c5bc662aa754f1095 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 00:20:01 +0800 Subject: [PATCH 69/88] Update api.config.js --- config/api.config.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/config/api.config.js b/config/api.config.js index 18f39f5ebe..236d8fc910 100755 --- a/config/api.config.js +++ b/config/api.config.js @@ -17,7 +17,7 @@ // throw new Error('`clientSecret` is not defined in api.config.js'); // } -module.exports = { module.exports = { +module.exports = { // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. // clientId: clientIdEnv, From 028bd1aa4e59daad8d96f1fff87f56ad78bb4be2 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 00:48:01 +0800 Subject: [PATCH 70/88] Update step-1.tsx --- src/pages/onedrive-vercel-index-oauth/step-1.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx index aedf7a79bb..c0902275bd 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -99,7 +99,7 @@ export default function OAuthStep1({ clientId, clientSecret }) { CLIENT_ID - {obfuscateSensitiveData(clientId)} + {clientId} @@ -107,7 +107,7 @@ export default function OAuthStep1({ clientId, clientSecret }) { CLIENT_SECRET* - {obfuscateSensitiveData(clientSecret)} + {clientSecret} From e1df3de76f1929c313d605b04579e2077931b6fd Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 00:51:54 +0800 Subject: [PATCH 71/88] Update step-3.tsx --- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index cd5756fb3e..28bce3a207 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -19,8 +19,8 @@ export async function getServerSideProps({ query, locale }) { const userPrincipalName = process.env.USER_PRINCIPAL_NAME || ''; // 检查是否已经通过OAuth认证 - const accessToken = await getAccessToken(); - if (accessToken) { + const existingAccessToken = await getAccessToken(); + if (existingAccessToken) { // 如果已经通过OAuth认证,重定向到主页 return { edirect: { From 1b48fdae00c65102471e0c3558dbdecff58b6989 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 10:46:30 +0800 Subject: [PATCH 72/88] Update index.ts --- src/pages/api/index.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/pages/api/index.ts b/src/pages/api/index.ts index 017fd0825b..06b1a5990f 100755 --- a/src/pages/api/index.ts +++ b/src/pages/api/index.ts @@ -11,6 +11,7 @@ import { getOdAuthTokens, storeOdAuthTokens } from '../../utils/odAuthTokenStore import { runCorsMiddleware } from './raw' const basePath = pathPosix.resolve('/', process.env.BASE_DIRECTORY || '/') +const clientId = process.env.CLIENT_ID || '' const clientSecret = revealObfuscatedToken(process.env.CLIENT_SECRET || '') /** @@ -50,7 +51,7 @@ export async function getAccessToken(): Promise { // Fetch new access token with in storage refresh token const body = new URLSearchParams() - body.append('client_id', process.env.CLIENT_ID || '') + body.append('client_id', clientId) body.append('redirect_uri', apiConfig.redirectUri) body.append('client_secret', clientSecret) body.append('refresh_token', refreshToken) From 7485dcac0272dbe68e14c3e640217db01482446d Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 12:22:37 +0800 Subject: [PATCH 73/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 64 +++++++++++++++++++++++---------------- 1 file changed, 38 insertions(+), 26 deletions(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 8197141db5..f80ce3d400 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -58,38 +58,50 @@ export function extractAuthCodeFromRedirected(url: string): string { // and returns the access token and refresh token on success. export async function requestTokenWithAuthCode( code: string + retry = 5 ): Promise< | { expiryTime: string; accessToken: string; refreshToken: string } | { error: string; errorDescription: string; errorUri: string } > { - const config = await getConfig() - const clientId = config.clientId - const clientSecret = revealObfuscatedToken(config.clientSecret) - const { redirectUri, authApi } = apiConfig + try { + const config = await getConfig() + const clientId = config.clientId + const clientSecret = revealObfuscatedToken(config.clientSecret) + const { redirectUri, authApi } = apiConfig - // Construct URL parameters for OAuth2 - const params = new URLSearchParams() - params.append('client_id', clientId) - params.append('redirect_uri', redirectUri) - params.append('client_secret', clientSecret) - params.append('code', code) - params.append('grant_type', 'authorization_code') + // Construct URL parameters for OAuth2 + const params = new URLSearchParams() + params.append('client_id', clientId) + params.append('redirect_uri', redirectUri) + params.append('client_secret', clientSecret) + params.append('code', code) + params.append('grant_type', 'authorization_code') - // Request access token - return axios - .post(authApi, params, { - headers: { - 'Content-Type': 'application/x-www-form-urlencoded', - }, - }) - .then(resp => { - const { expires_in, access_token, refresh_token } = resp.data - return { expiryTime: expires_in, accessToken: access_token, refreshToken: refresh_token } - }) - .catch(err => { - const { error, error_description, error_uri } = err.response.data - return { error, errorDescription: error_description, errorUri: error_uri } - }) + // Request access token + return axios + .post(authApi, params, { + headers: { + 'Content-Type': 'application/x-www-form-urlencoded', + }, + }) + .then(resp => { + const { expires_in, access_token, refresh_token } = resp.data + return { expiryTime: expires_in, accessToken: access_token, refreshToken: refresh_token } + }) + .catch(err => { + const { error, error_description, error_uri } = err.response.data + return { error, errorDescription: error_description, errorUri: error_uri } + }) + } catch (error) { + console.error("Failed to get api/config:", error) + if (retry > 0) { + console.log(`Retrying... ${retry} attempts left.`) + return requestTokenWithAuthCode(code, retry - 1) + } + else { + return { error: "Failed to get api/config", errorDescription: error.message } + } + } } // Verify the identity of the user with the access token and compare it with the userPrincipalName From b5fa5575332365a72da58af0e81469b1f69fc385 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 12:23:56 +0800 Subject: [PATCH 74/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index f80ce3d400..4457b40add 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -57,7 +57,7 @@ export function extractAuthCodeFromRedirected(url: string): string { // will be used to request an access token. This function requests the access token with the authorisation code // and returns the access token and refresh token on success. export async function requestTokenWithAuthCode( - code: string + code: string, retry = 5 ): Promise< | { expiryTime: string; accessToken: string; refreshToken: string } From 4eb8621dac2ddc03174e69c0a12ee8dcb4b6f000 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 12:30:48 +0800 Subject: [PATCH 75/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 4457b40add..1814bb8b65 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -99,7 +99,7 @@ export async function requestTokenWithAuthCode( return requestTokenWithAuthCode(code, retry - 1) } else { - return { error: "Failed to get api/config", errorDescription: error.message } + return { error: "Failed to get api/config", errorDescription: error.message, errorUri: "" } } } } From 25a73cca248b7fa249a7c7d214bcae8b9dafd513 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 12:34:35 +0800 Subject: [PATCH 76/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 1814bb8b65..f95dfaf885 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -92,14 +92,18 @@ export async function requestTokenWithAuthCode( const { error, error_description, error_uri } = err.response.data return { error, errorDescription: error_description, errorUri: error_uri } }) - } catch (error) { - console.error("Failed to get api/config:", error) + } catch (error: unknown) { + console.error("Failed to get config:", error) + let errorMessage = "" + if (error instanceof Error) { + errorMessage = error.message + } if (retry > 0) { console.log(`Retrying... ${retry} attempts left.`) return requestTokenWithAuthCode(code, retry - 1) } else { - return { error: "Failed to get api/config", errorDescription: error.message, errorUri: "" } + return { error: "Failed to get config", errorDescription: error.message, errorUri: "" } } } } From 29f812a7499ac908091a03c1b4a12eeaef1cd9dd Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 12:42:52 +0800 Subject: [PATCH 77/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index f95dfaf885..a518acab4a 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -92,7 +92,8 @@ export async function requestTokenWithAuthCode( const { error, error_description, error_uri } = err.response.data return { error, errorDescription: error_description, errorUri: error_uri } }) - } catch (error: unknown) { + } + catch (error) { console.error("Failed to get config:", error) let errorMessage = "" if (error instanceof Error) { @@ -103,10 +104,15 @@ export async function requestTokenWithAuthCode( return requestTokenWithAuthCode(code, retry - 1) } else { - return { error: "Failed to get config", errorDescription: error.message, errorUri: "" } + if (error instanceof Error) { + return { error: "Failed to get config", errorDescription: error.message, errorUri: "" } + } + else { + // If error is not an instance of Error, we can return a generic error message + return { error: "Failed to get config", errorDescription: "Unknown error", errorUri: "" } + } } } -} // Verify the identity of the user with the access token and compare it with the userPrincipalName // in the Microsoft Graph API. If the userPrincipalName matches, proceed with token storing. From 8f9a294f67ae4bb6351aa11ce3798f33d659b146 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 12:48:42 +0800 Subject: [PATCH 78/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index a518acab4a..1fac3729b3 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -113,6 +113,7 @@ export async function requestTokenWithAuthCode( } } } +} // Verify the identity of the user with the access token and compare it with the userPrincipalName // in the Microsoft Graph API. If the userPrincipalName matches, proceed with token storing. From 94cd9d9e8a048608d7b9569cdd65e659c9c791e8 Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Tue, 11 Jul 2023 16:32:43 +0800 Subject: [PATCH 79/88] Update oAuthHandler.ts --- src/utils/oAuthHandler.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 1fac3729b3..58252330e7 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -24,8 +24,7 @@ export function revealObfuscatedToken(obfuscated: string): string { // Generate the Microsoft OAuth 2.0 authorization URL, used for requesting the authorisation code export async function generateAuthorisationUrl(): Promise { - const config = await getConfig() - const clientId = config.clientId + const { clientId } = await getConfig() const { redirectUri, authApi, scope } = apiConfig const authUrl = authApi.replace('/token', '/authorize') From c7c163f4a2535bf67f16ceb538615f593e6e67a3 Mon Sep 17 00:00:00 2001 From: eks Date: Wed, 12 Jul 2023 10:36:38 +0800 Subject: [PATCH 80/88] update --- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 8 +++++++- src/utils/oAuthHandler.ts | 2 +- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 28bce3a207..6493e5e676 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -10,12 +10,14 @@ import siteConfig from '../../../config/site.config' import Navbar from '../../components/Navbar' import Footer from '../../components/Footer' -import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer } from '../../utils/oAuthHandler' +import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer, getConfig } from '../../utils/oAuthHandler' import { LoadingIcon } from '../../components/Loading' import { getAccessToken } from '../api' // 导入getAccessToken函数 export async function getServerSideProps({ query, locale }) { const { authCode } = query + const clientId = process.env.CLIENT_ID || ''; + const clientSecret = process.env.CLIENT_SECRET || ''; const userPrincipalName = process.env.USER_PRINCIPAL_NAME || ''; // 检查是否已经通过OAuth认证 @@ -37,6 +39,8 @@ export async function getServerSideProps({ query, locale }) { error: 'No auth code present', description: 'Where is the auth code? Did you follow step 2 you silly donut?', ...(await serverSideTranslations(locale, ['common'])), + clientId, + clientSecret, userPrincipalName, }, } @@ -60,6 +64,8 @@ export async function getServerSideProps({ query, locale }) { return { props: { + clientId, + clientSecret, userPrincipalName, error: null, expiryTime, diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 58252330e7..53f8dc5565 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -57,13 +57,13 @@ export function extractAuthCodeFromRedirected(url: string): string { // and returns the access token and refresh token on success. export async function requestTokenWithAuthCode( code: string, + config: any, retry = 5 ): Promise< | { expiryTime: string; accessToken: string; refreshToken: string } | { error: string; errorDescription: string; errorUri: string } > { try { - const config = await getConfig() const clientId = config.clientId const clientSecret = revealObfuscatedToken(config.clientSecret) const { redirectUri, authApi } = apiConfig From 8df36a7e92b50bd8e3dd9f7bedddabae48225f81 Mon Sep 17 00:00:00 2001 From: eks Date: Wed, 12 Jul 2023 10:39:59 +0800 Subject: [PATCH 81/88] Update step-3.tsx --- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 6493e5e676..50121c21fb 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -10,7 +10,7 @@ import siteConfig from '../../../config/site.config' import Navbar from '../../components/Navbar' import Footer from '../../components/Footer' -import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer, getConfig } from '../../utils/oAuthHandler' +import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer } from '../../utils/oAuthHandler' import { LoadingIcon } from '../../components/Loading' import { getAccessToken } from '../api' // 导入getAccessToken函数 From 6a838f4f80544f22788235e161b455782a96d5fa Mon Sep 17 00:00:00 2001 From: eks Date: Wed, 12 Jul 2023 10:49:19 +0800 Subject: [PATCH 82/88] Update step-3.tsx --- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 50121c21fb..ddbb0b362b 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -45,8 +45,8 @@ export async function getServerSideProps({ query, locale }) { }, } } - - const response = await requestTokenWithAuthCode(authCode) + const config = { clientId, clientSecret, userPrincipalName, baseDirectory }; + const response = await requestTokenWithAuthCode(authCode, config) // If error response, return invalid if ('error' in response) { From 3296d666e38cc7a23a328a77aff65770a9ab1e10 Mon Sep 17 00:00:00 2001 From: eks Date: Wed, 12 Jul 2023 10:50:28 +0800 Subject: [PATCH 83/88] Update step-3.tsx --- src/pages/onedrive-vercel-index-oauth/step-3.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index ddbb0b362b..4ddda11c08 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -45,7 +45,7 @@ export async function getServerSideProps({ query, locale }) { }, } } - const config = { clientId, clientSecret, userPrincipalName, baseDirectory }; + const config = { clientId, clientSecret, userPrincipalName }; const response = await requestTokenWithAuthCode(authCode, config) // If error response, return invalid From 5f06ef74c905f272b3ae355add642f6164355ade Mon Sep 17 00:00:00 2001 From: eks <134615514+ailoha@users.noreply.github.com> Date: Wed, 12 Jul 2023 14:43:43 +0800 Subject: [PATCH 84/88] Update TEST-BUTTON.md --- TEST-BUTTON.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/TEST-BUTTON.md b/TEST-BUTTON.md index ad5fbc417c..a35911d8f9 100644 --- a/TEST-BUTTON.md +++ b/TEST-BUTTON.md @@ -1 +1 @@ -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Failoha%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Failoha%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) From 46a64c5ca89b1799b34cee156afd12c39633c09a Mon Sep 17 00:00:00 2001 From: eks Date: Wed, 12 Jul 2023 16:34:57 +0800 Subject: [PATCH 85/88] Delete TEST-BUTTON.md --- TEST-BUTTON.md | 1 - 1 file changed, 1 deletion(-) delete mode 100644 TEST-BUTTON.md diff --git a/TEST-BUTTON.md b/TEST-BUTTON.md deleted file mode 100644 index a35911d8f9..0000000000 --- a/TEST-BUTTON.md +++ /dev/null @@ -1 +0,0 @@ -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2Failoha%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) From 4e145057dd38d7f3388fa5f5f1e250c77c4c9b95 Mon Sep 17 00:00:00 2001 From: eks Date: Wed, 12 Jul 2023 16:42:31 +0800 Subject: [PATCH 86/88] update README --- README.md | 22 +++++++++------------- README.zh-CN.md | 24 ++++++++++-------------- 2 files changed, 19 insertions(+), 27 deletions(-) diff --git a/README.md b/README.md index 568894640c..b4eb657235 100644 --- a/README.md +++ b/README.md @@ -34,19 +34,19 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h **Once you're prepared, you can click the button below to deploy:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET) > - If you have folders that need password protection. > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) with `NEXT_PUBLIC_PROTECTED_ROUTES` +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET) with `NEXT_PUBLIC_PROTECTED_ROUTES` > > - If you have multiple OneDrive accounts that need to use the same Redis database. > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` > > - If you need to deploy multiple OneDrive-Index, and all have folders that need password protection. > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` - After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. @@ -62,10 +62,10 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h | Name | Description | Original Path | Note | | --- | --- | --- | --- | | `NEXT_PUBLIC_SITE_TITLE` | Title of the display page | `config/site.config.js` | e.g. Nicaragua's richest man's OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | Your OneDrive account | `config/site.config.js` | **Case-sensitive** | -| `NEXT_PUBLIC_BASE_DIRECTORY` | The OneDrive directory you want to share | `config/site.config.js` | `/directory name`, root directory is `/` | -| `NEXT_PUBLIC_CLIENT_ID` | The client ID of the app you registered in Microsoft Azure | `config/api.config.js` | The one provided by the original author has expired, it is recommended to register one yourself, the validity period can be set to two years (anyway, you have to set the API permissions of the account, by the way). The acquisition method refers to the [DOCS](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) | -| `NEXT_PUBLIC_CLIENT_SECRET` | The client secret of the app registered in Microsoft Azure | `config/api.config.js` | The acquisition method is the same, especially note that this **needs to encrypt the original secret with AES** (can be done in the [DOCS](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | +| `USER_PRINCIPAL_NAME` | Your OneDrive account | `config/site.config.js` | **Case-sensitive** | +| `BASE_DIRECTORY` | The OneDrive directory you want to share | `config/site.config.js` | `/directory name`, root directory is `/` | +| `CLIENT_ID` | The client ID of the app you registered in Microsoft Azure | `config/api.config.js` | The one provided by the original author has expired, it is recommended to register one yourself, the validity period can be set to two years (anyway, you have to set the API permissions of the account, by the way). The acquisition method refers to the [DOCS](https://ovi.swo.moe/docs/advanced#using-your-own-clientid-and-clientsecret) | +| `CLIENT_SECRET` | The client secret of the app registered in Microsoft Azure | `config/api.config.js` | The acquisition method is the same, especially note that this **needs to encrypt the original secret with AES** (can be done in the [DOCS](https://ovi.swo.moe/docs/advanced#modify-configs-in-apiconfigjs)) | ### Optional Variables | Name | Description | Original Path | Note | @@ -80,7 +80,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ## Security Risks -- In both this version and the original archived version, the deployer's OneDrive account `USER_PRINCIPAL_NAME` is leaked in the source code of the webpage. +- In the original archived version, the deployer's OneDrive account `USER_PRINCIPAL_NAME` is leaked in the source code of the webpage. - In the original archived version, the deployer's `clientId` can be seen in the link used to obtain the authorization code in OAuth Step-2. The `obfuscatedClientSecret` can be seen in the source code of the OAuth Step-1. @@ -88,14 +88,10 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - Because of the design decision of Next.js, environment variables starting with `NEXT_PUBLIC_` are not only available on the server side, but also on the client side (browser). This means that any environment variable starting with `NEXT_PUBLIC_` will be included in the built JavaScript file and will be sent to the user's browser. Therefore, anyone visiting your website can view the values of these environment variables by viewing the source code of the website or network requests. -- At the beginning, when setting the `clientId` and `obfuscatedClientSecret` in `config/api.config.js` to the environment variables, I have tried to use environment variable key names that do not start with `NEXT_PUBLIC_`, but there will be various problems during OAuth authentication and OAuth authentication cannot be completed. In order to deploy smoothly, I had to use environment variable key names starting with `NEXT_PUBLIC_`. Considering that `clientId` and `ClientSecret` will not have much problem without the login password of the OneDrive account, and it is also set that the OAuth authentication page cannot be accessed easily to obtain these sensitive information after completing OAuth authentication, it is temporarily solved in this way. - ## Todo List - Put the password in the environment variables instead of the `.password` file. -- Deepen the study of the original version of the code and strive to implement the function with environment variable key names that do not start with `NEXT_PUBLIC_`, to improve security. - - Redesign the LOGO. The contrast of the original LOGO is too low, and it is not consistent enough with the style of other icons and fonts on the page. ## License diff --git a/README.zh-CN.md b/README.zh-CN.md index fb407f6870..a8538fc758 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -34,19 +34,19 @@ **当您做好准备工作,就可以点击下面的按钮进行部署了:** -[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET) > - 如果你有需要密码保护的目录: > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET) with `NEXT_PUBLIC_PROTECTED_ROUTES` +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET) with `NEXT_PUBLIC_PROTECTED_ROUTES` > > - 如果你有多个OneDrive帐户需要使用同一个Redis数据库: > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `KV_PREFIX` > > - 如果你需要部署多个OneDrive-Index同时都有需要密码保护的目录: > -> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,NEXT_PUBLIC_USER_PRINCIPAL_NAME,NEXT_PUBLIC_BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,NEXT_PUBLIC_CLIENT_ID,NEXT_PUBLIC_CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` +> [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` - 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 @@ -62,10 +62,10 @@ | 名称 | 描述 | 原路径 | 说明 | | --- | --- | --- | --- | | `NEXT_PUBLIC_SITE_TITLE` | 展示页面的标题 | `config/site.config.js` | 例如:尼加拉瓜首富的OneDrive | -| `NEXT_PUBLIC_USER_PRINCIPAL_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | -| `NEXT_PUBLIC_BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | -| `NEXT_PUBLIC_CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | -| `NEXT_PUBLIC_CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | +| `USER_PRINCIPAL_NAME` | 您的OneDrive帐户 | `config/site.config.js` | **字母大小写必须一致** | +| `BASE_DIRECTORY` | 您要分享的OneDrive目录 | `config/site.config.js` | (格式为`/目录名`),根目录则填写`/` | +| `CLIENT_ID` | 您在微软Azure注册的应用程序客户端ID | `config/api.config.js` | 原作者提供的已过期,建议自己注册一个,有效期可以设到两年(反正也要设置帐户的API权限,顺道咯)。获取方式参照原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#使用你自己的-client-id-与-secret) | +| `CLIENT_SECRET` | 您在微软Azure注册的应用程序客户端密钥 | `config/api.config.js` | 获取方式同上,特别注意这个**需要对原密钥进行AES加密**(可在原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-apiconfigjs)中进行) | ### 可选变量 | 名称 | 描述 | 原路径 | 说明 | @@ -80,22 +80,18 @@ ## 安全风险 -- 本版本和原作者的存档版本中,部署者的OneDrive账户`USER_PRINCIPAL_NAME`都会暴露在网页的源代码中。 +- 原作者的存档版本中,部署者的OneDrive账户`USER_PRINCIPAL_NAME`都会暴露在网页的源代码中。 - 在原作者的存档版本中,部署者的`clientId`可以在OAuth认证第二步中获取授权码的链接中看到,`obfuscatedClientSecret`可以在OAuth认证第一步的源代码中看到。 -> 本版本在执行OAuth认证过程时会检查是否已经通过了认证,若已通过认证则会重定向到主页,否则才会继续进行OAuth认证过程。如此试图阻止有心人通过OAuth认证的链接地址获取`clientId`和`obfuscatedClientSecret`的值。 +> 本版本在执行OAuth认证过程时会检查是否已经通过了认证,若已通过认证则会重定向到主页,否则才会继续进行OAuth认证过程。 - 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 -- 最开始在把`config/api.config.js`中的`clientId`和`obfuscatedClientSecret`放在环境变量中设置时,有试过使用不以`NEXT_PUBLIC_`开头的环境变量键名,但会在OAuth认证时,出现各种各样的问题而无法完成OAuth认证。为了顺利部署,只好先使用以`NEXT_PUBLIC_`开头的环境变量键名了。考虑到`clientId`和`ClientSecret`在没有OneDrive帐户的登录密码时,也不会有太大问题,并且也设定了当完成OAuth认证后无法轻易访问OAuth认证页面获取这些敏感信息,就暂时这样解决了。 - ## 待办事项 - 将密码放在环境变量中,而不是加密目录下的`.password`文件中。 -- 深入研究原始版本的代码,努力以非`NEXT_PUBLIC_`开头的环境变量键名实现功能,以提高安全性。 - - 重新设计LOGO。原始LOGO的对比度太低,与页面上其他图标和字体的风格不够一致。 ## License From f54ca5dee86490437b3754a8be3fc460dcd36778 Mon Sep 17 00:00:00 2001 From: eks Date: Thu, 13 Jul 2023 11:24:59 +0800 Subject: [PATCH 87/88] Refactor code and update comments for clarity --- README.md | 32 +++++----- README.zh-CN.md | 37 +++++++----- config/api.config.js | 16 +---- config/site.config.js | 60 +++++++++---------- .../onedrive-vercel-index-oauth/step-1.tsx | 11 ++-- .../onedrive-vercel-index-oauth/step-2.tsx | 9 +-- .../onedrive-vercel-index-oauth/step-3.tsx | 17 +++--- src/utils/oAuthHandler.ts | 28 ++------- 8 files changed, 91 insertions(+), 119 deletions(-) diff --git a/README.md b/README.md index b4eb657235..e277ce18ab 100644 --- a/README.md +++ b/README.md @@ -6,17 +6,19 @@ This project is a fork from [spencerwooo/onedrive-vercel-index](https://github.c > This version has only been tested with an E5 Developer account. Other types of OneDrive accounts need further testing. -## Demo +## Modifications -The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](https://odi-demo.freeloop.one) of this One-Click Deployment version. +- In this version, some variables that needed to be set in the `api.config.js` and `site.config.js` configuration files in the `config/` are now set in the environment variables of Vercel. In this way, there is no need to - first fork the original repository - then manually modify the configuration file - and then deploy. Instead, you can directly click the one-click deployment button in this document, enter the values of the environment variables during the deployment process, and then complete the deployment. -![demo](./public/demo.png) +> In this version, some sensitive variables are set using environment variables with prefixes other than `NEXT_PUBLIC_`. This is done to prevent casual website visitors from easily obtaining your OneDrive account, ClientID, and ClientSecret information. -## Modifications +- Additionally, this version is set to automatically close the OAuth authentication channel after OAuth authentication is completed. This is to prevent malicious individuals from easily obtaining user configuration information through the OAuth authentication URL link. + +## Demo -- This version mainly moves some parameters that need to be set in the `api.config.js` and `site.config.js` in the `config/` to the environment variables of Vercel for setting. In this way, there is no need to - first fork the original repository - then manually modify the configuration file - and then deploy, but you can directly click the one-click deployment button in this document, enter the value of the environment variable during the deployment process, and then complete the deployment. +The [Demo](https://odi-demo.freeloop.one) of this One-Click Deploy version. | The [Demo](https://drive.swo.moe) (UNMAINTAINED) by the original author. -- Another thing is that this version is set to automatically close the OAuth authentication channel after completing OAuth authentication, to prevent people with intentions from easily obtaining user configuration information through the OAuth authentication URL link. +![demo](./public/demo.png) ## Getting Started @@ -26,13 +28,13 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h - This project retrieves the file list and download links by calling OneDrive's API, so setting up the API permissions for your OneDrive account is essential. Please refer to the [DOCS](https://ovi.swo.moe/docs/advanced#modify-api-permissions). -- The three API permissions that need to be set up are: `user.read`, `files.read.all`, `offline_access`. +> The three API permissions that need to be set up are: `user.read`, `files.read.all`, `offline_access`. -2. **Prepare the five [necessary environmental variables](#necessary-variables) to be filled in during deployment on Vercel.** +2. **Prepare the five [necessary environmental variables (click to view)](#necessary-variables) to be filled in during deployment on Vercel.** ### Deploying to Vercel -**Once you're prepared, you can click the button below to deploy:** +3. **Once you're prepared, you can click the button below to deploy:** [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET) @@ -48,7 +50,7 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h > > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` -- After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. +4. After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. > `REDIS_URL`:If you are encountering Redis database for the first time, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash)(simply create a new database in the `Redis` of Upstash, then create a new integration in `Vercel Integrations`, and associate the just deployed OneDrive-Index project with the Redis database), it will automatically fill in the environment variables after project deployment. @@ -80,18 +82,20 @@ The [Demo](https://drive.swo.moe) provided by the original author | The [Demo](h ## Security Risks -- In the original archived version, the deployer's OneDrive account `USER_PRINCIPAL_NAME` is leaked in the source code of the webpage. - -- In the original archived version, the deployer's `clientId` can be seen in the link used to obtain the authorization code in OAuth Step-2. The `obfuscatedClientSecret` can be seen in the source code of the OAuth Step-1. +- In the archived version of the original author, the `userPrincipalName`, `clientId`, and `obfuscatedClientSecret` of the OneDrive account of the deployer are exposed in the source code of the web page. -> This version checks whether authentication has already been passed when performing the OAuth authentication process. If it has, it redirects to the homepage; otherwise, it proceeds with the OAuth authentication process. It attempts to prevent individuals with malicious intent from obtaining the values of `clientId` and `obfuscatedClientSecret` through the link address of OAuth authentication. +> This version checks whether authentication has already been passed when performing the OAuth authentication process. If it has, it redirects to the homepage, otherwise, it proceeds with the OAuth authentication process. It attempts to prevent individuals with malicious intent from obtaining the values of `clientId` and `obfuscatedClientSecret` through the link address of OAuth authentication. - Because of the design decision of Next.js, environment variables starting with `NEXT_PUBLIC_` are not only available on the server side, but also on the client side (browser). This means that any environment variable starting with `NEXT_PUBLIC_` will be included in the built JavaScript file and will be sent to the user's browser. Therefore, anyone visiting your website can view the values of these environment variables by viewing the source code of the website or network requests. +> This version uses non-`NEXT_PUBLIC_` prefixed environment variables for the `userPrincipalName`, `clientId`, `obfuscatedClientSecret`, and `baseDirectory` variables, making it as difficult as possible for website visitors to easily obtain your OneDrive account, ClientID, and ClientSecret information. + ## Todo List - Put the password in the environment variables instead of the `.password` file. +> However, in this way, it is more difficult to set different access passwords for different encrypted directories. + - Redesign the LOGO. The contrast of the original LOGO is too low, and it is not consistent enough with the style of other icons and fonts on the page. ## License diff --git a/README.zh-CN.md b/README.zh-CN.md index a8538fc758..5fe82e4503 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -6,17 +6,19 @@ > 本版本只测试通过E5开发者帐户,其他类型的OneDrive帐户有待进一步测试。 -## 在线预览 +## 修改说明 -原作者提供的[在线预览](https://drive.swo.moe) | 本一键部署版的[在线预览](https://odi-demo.freeloop.one) +- 本版本主要把原本需要在`config/`目录下的`api.config.js`和`site.config.js`这两个配置文件中设置的一些变量搬到了Vercel的环境变量中进行设置。如此便无须——先fork原仓库——然后手动修改配置文件——再部署,而是可以直接点击本文档中的一键部署按钮,在部署过程中输入环境变量的值,然后完成部署。 -![demo](./public/demo.png) +> 本版本把一些比较敏感的变量使用了非`NEXT_PUBLIC_`前缀的环境变量进行设置,尽可能让网站浏览者不能轻易获取你的OneDrive帐号、ClientID以及ClientSecret等信息。 -## 修改说明 +- 再就是本版本设定了当完成OAuth认证后,自动关闭OAuth认证通道,以防有心人通过OAuth认证的网址链接就轻易地获取到用户的配置信息。 -- 本版本主要把原本需要在`config/`目录下的`api.config.js`和`site.config.js`这两个配置文件中设置的一些参数搬到了Vercel的环境变量中进行设置,如此便无须——先fork原仓库——然后手动修改配置文件——再部署,而是可以直接点击本文档中的一键部署按钮,在部署过程中输入环境变量的值,然后完成部署。 +## 在线预览 -- 再就是本版本设定了当完成OAuth认证后,自动关闭OAuth认证通道,以防有心人通过OAuth认证的网址链接就轻易地获取到用户的配置信息。 +本一键部署版的[在线预览](https://odi-demo.freeloop.one) | 原作者提供的[在线预览](https://drive.swo.moe)(已暂停维护) + +![demo](./public/demo.png) ## 部署方法 @@ -26,13 +28,13 @@ - 本项目是通过调用OneDrive的API来获取文件列表以及下载链接的,所以设置OneDrive帐户的API权限是必须的,获取方法请参考原作者编写的[使用文档](https://ovi.swo.moe/zh/docs/advanced#修改-api-权限)。 -- **需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。** +> **需要设置的API权限为以下三个:`user.read`、`files.read.all`、`offline_access`。** -2. **准备好在Vercel部署时填写的五个[必要环境变量](#必要参数):** +2. **准备好在Vercel部署时填写的五个[必要环境变量(点击查看说明)](#必要变量)的值:** ### 部署到Vercel -**当您做好准备工作,就可以点击下面的按钮进行部署了:** +3. **当您做好准备工作,就可以点击下面的按钮进行部署了:** [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,CLIENT_ID,CLIENT_SECRET) @@ -48,13 +50,13 @@ > > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` -- 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 +4. 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 > `REDIS_URL`:如果您是第一次接触Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 -- `REDIS_URL`设置成功后,再重新部署一次项目。 +5. `REDIS_URL`设置成功后,再重新部署一次项目。 -**部署成功后,当您第一次访问您的`onedrive-vercel-index`页面时,会引导你进行OAuth认证(相当简单),详情请参考原作者编写的[说明文档](https://ovi.swo.moe/zh/docs/getting-started#进行认证)。** +6. **部署成功后,当您第一次访问您的`onedrive-vercel-index`页面时,会引导你进行OAuth认证(相当简单),详情请参考原作者编写的[说明文档](https://ovi.swo.moe/zh/docs/getting-started#进行认证)。** ## 环境变量 @@ -80,18 +82,21 @@ ## 安全风险 -- 原作者的存档版本中,部署者的OneDrive账户`USER_PRINCIPAL_NAME`都会暴露在网页的源代码中。 +- 在原作者的存档版本中,部署者的OneDrive账户`userPrincipalName`、`clientId`和`obfuscatedClientSecret`都会暴露在网页的源代码中。 -- 在原作者的存档版本中,部署者的`clientId`可以在OAuth认证第二步中获取授权码的链接中看到,`obfuscatedClientSecret`可以在OAuth认证第一步的源代码中看到。 +> 本版本在执行OAuth认证过程时会检查是否已经通过了认证,若已通过认证则会重定向到主页,否则才会继续进行OAuth认证过程。如此访问者便不能通过访问OAuth认证的链接地址轻易获取你的`clientId` +和`clientSecret`了。 -> 本版本在执行OAuth认证过程时会检查是否已经通过了认证,若已通过认证则会重定向到主页,否则才会继续进行OAuth认证过程。 +- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何`NEXT_PUBLIC_`前缀的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 -- 因为Next.js的设计决策,以`NEXT_PUBLIC_`开头的环境变量不仅在服务器端可用,而且在客户端(浏览器)也可用。这意味着任何以`NEXT_PUBLIC_`开头的环境变量都会被包含在构建的JavaScript文件中,并会被发送到用户的浏览器。因此,任何访问你的网站的人都可以通过查看网站的源代码或网络请求来查看这些环境变量的值。 +> 本版本把`userPrincipalName`、`clientId`、`obfuscatedClientSecret`以及`baseDirectory`变量使用了非`NEXT_PUBLIC_`前缀的环境变量,尽可能让网站浏览者不能轻易获取你的OneDrive帐号、ClientID以及ClientSecret等信息。 ## 待办事项 - 将密码放在环境变量中,而不是加密目录下的`.password`文件中。 +> 不过如此便比较难为不同的加密目录设置不同的访问密码了。 + - 重新设计LOGO。原始LOGO的对比度太低,与页面上其他图标和字体的风格不够一致。 ## License diff --git a/config/api.config.js b/config/api.config.js index 236d8fc910..ed8732a3f0 100755 --- a/config/api.config.js +++ b/config/api.config.js @@ -8,20 +8,10 @@ * In which case you would need to change directLinkRegex. */ -// const clientIdEnv = process.env.NEXT_PUBLIC_CLIENT_ID; -// if (!clientIdEnv) { -// throw new Error('`clientId` is not defined in api.config.js'); -// } -// const clientSecretEnv = process.env.NEXT_PUBLIC_CLIENT_SECRET; -// if (!clientSecretEnv) { -// throw new Error('`clientSecret` is not defined in api.config.js'); -// } - module.exports = { - // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. You would - // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. // not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. -// clientId: clientIdEnv, -// obfuscatedClientSecret: clientSecretEnv, + // The clientId and clientSecret are used to authenticate the user with Microsoft Graph API using OAuth. + // If you are an E5 developer subscriber, set them in Vercel's environment variables `CLIENT_ID` and `CLIENT_SECRET` + // You would not need to change anything here if you can authenticate with your personal Microsoft account with OneDrive International. // The redirectUri is the URL that the user will be redirected to after they have authenticated with Microsoft Graph API. // Likewise, you would not need to change redirectUri if you are using your personal Microsoft account with OneDrive International. diff --git a/config/site.config.js b/config/site.config.js index c5a173de44..a9a97d0da5 100755 --- a/config/site.config.js +++ b/config/site.config.js @@ -3,24 +3,40 @@ * the title, used Google fonts, site icons, contact info, etc. */ module.exports = { - // This is what we use to identify who you are when you are initialising the website for the first time. - // Make sure this is exactly the same as the email address you use to sign into your Microsoft account. - // You can also put this in your Vercel's environment variable 'NEXT_PUBLIC_USER_PRINCIPLE_NAME' if you worry about - // your email being exposed in public. - // userPrincipalName: process.env.NEXT_PUBLIC_USER_PRINCIPAL_NAME || '', - - // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. It should be placed under the - // /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. + // [OPTIONAL] This is the website icon to the left of the title inside the navigation bar. + // It should be placed under the /public directory of your GitHub project (not your OneDrive folder!), and referenced here by its relative path to /public. icon: '/icons/128.png', - // Prefix for KV Storage + // Prefix for KV Storage. + // You can put this in Vercel's environment variable 'KV_PREFIX' without any modification here. kvPrefix: process.env.KV_PREFIX || '', // The name of your website. Present alongside your icon. + // You can put this in Vercel's environment variable 'NEXT_PUBLIC_SITE_TITLE' without any modification here. title: process.env.NEXT_PUBLIC_SITE_TITLE || 'OneDrive-Vercel-Index', - // The folder that you are to share publicly with onedrive-vercel-index. Use '/' if you want to share your root folder. - // baseDirectory: process.env.NEXT_PUBLIC_BASE_DIRECTORY || '/', + // [OPTIONAL] This is where you specify the folders that are password protected. + // It is an array of paths pointing to all the directories in which you have .password set. Check the documentation for details. + // You can put this in Vercel's environment variable 'NEXT_PUBLIC_PROTECTED_ROUTES' without any modification here. + protectedRoutes: process.env.NEXT_PUBLIC_PROTECTED_ROUTES ? process.env.NEXT_PUBLIC_PROTECTED_ROUTES.split(',') : [], + + // [OPTIONAL] If you want to display the email used to contact you on the right side of the nav bar, + // you can set it in Vercel's environment variable 'NEXT_PUBLIC_EMAIL' without any modification here. + email: process.env.NEXT_PUBLIC_EMAIL ? `mailto:${process.env.NEXT_PUBLIC_EMAIL}` : '', + + // [OPTIONAL] The footer component of your website. You can write HTML here, but you need to escape double quotes - changing " to \". + // You can write anything here, and if you like badges, generate some with https://shields.io + footer: + 'Powered by
onedrive-vercel-index.', + + // [OPTIONAL] This is an array of names and links for setting your social information and links. + // In the latest update, all brand icons inside font awesome is supported and the icon to render is based on the name you provide. See the documentation for details. + links: [ + { + name: 'GitHub', + link: 'https://github.com/iRedScarf/onedrive-vercel-index', + }, + ], // [OPTIONAL] This represents the maximum number of items that one directory lists, pagination supported. // Do note that this is limited up to 200 items by the upstream OneDrive API. @@ -35,28 +51,6 @@ module.exports = { // googleFontLinks - an array of links for referencing the google font assets. googleFontLinks: ['https://fonts.googleapis.com/css2?family=Fira+Mono&family=Inter:wght@400;500;700&display=swap'], - // [OPTIONAL] The footer component of your website. You can write HTML here, but you need to escape double - // quotes - changing " to \". You can write anything here, and if you like badges, generate some with https://shields.io - footer: - 'Powered by onedrive-vercel-index.', - - // [OPTIONAL] This is where you specify the folders that are password protected. It is an array of paths pointing to all - // the directories in which you have .password set. Check the documentation for details. - protectedRoutes: process.env.NEXT_PUBLIC_PROTECTED_ROUTES ? process.env.NEXT_PUBLIC_PROTECTED_ROUTES.split(',') : [], - - // [OPTIONAL] Use "" here if you want to remove this email address from the nav bar. - email: process.env.NEXT_PUBLIC_EMAIL ? `mailto:${process.env.NEXT_PUBLIC_EMAIL}` : '', - - // [OPTIONAL] This is an array of names and links for setting your social information and links. - // In the latest update, all brand icons inside font awesome is supported and the icon to render is based on the name - // you provide. See the documentation for details. - links: [ - { - name: 'GitHub', - link: 'https://github.com/iRedScarf/onedrive-vercel-index', - }, - ], - // This is a day.js-style datetime format string to format datetimes in the app. Ref to // https://day.js.org/docs/en/display/format for detailed specification. The default value is ISO 8601 full datetime // without timezone and replacing T with space. diff --git a/src/pages/onedrive-vercel-index-oauth/step-1.tsx b/src/pages/onedrive-vercel-index-oauth/step-1.tsx index c0902275bd..f4c0d79e3d 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-1.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-1.tsx @@ -9,14 +9,14 @@ import apiConfig from '../../../config/api.config' import Navbar from '../../components/Navbar' import Footer from '../../components/Footer' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' -import { getAccessToken } from '../api' // 导入getAccessToken函数 +import { getAccessToken } from '../api' export async function getServerSideProps({ locale }) { const clientId = process.env.CLIENT_ID || ''; const clientSecret = process.env.CLIENT_SECRET || ''; - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - - // 如果访问令牌存在,重定向到主页 + // Get accessToken using getAccessToken function + const accessToken = await getAccessToken(); + // If the accessToken exists, redirect to the home page if (accessToken) { return { redirect: { @@ -25,8 +25,7 @@ export async function getServerSideProps({ locale }) { }, } } - - // 如果访问令牌不存在,正常渲染页面 + // If the accessToken does not exist, render the page normally return { props: { ...(await serverSideTranslations(locale, ['common'])), diff --git a/src/pages/onedrive-vercel-index-oauth/step-2.tsx b/src/pages/onedrive-vercel-index-oauth/step-2.tsx index 6782223e8c..e869ca72cb 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-2.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-2.tsx @@ -11,11 +11,12 @@ import Navbar from '../../components/Navbar' import Footer from '../../components/Footer' import { LoadingIcon } from '../../components/Loading' import { extractAuthCodeFromRedirected, generateAuthorisationUrl } from '../../utils/oAuthHandler' -import { getAccessToken } from '../api' // 导入getAccessToken函数 +import { getAccessToken } from '../api' export async function getServerSideProps({ locale }) { - const accessToken = await getAccessToken(); // 使用getAccessToken函数获取访问令牌 - // 如果访问令牌存在,重定向到主页 + // Get accessToken using getAccessToken function + const accessToken = await getAccessToken(); + // If the accessToken exists, redirect to the home page if (accessToken) { return { redirect: { @@ -24,7 +25,7 @@ export async function getServerSideProps({ locale }) { }, } } - // 如果访问令牌不存在,正常渲染页面 + // If the accessToken does not exist, render the page normally return { props: { ...(await serverSideTranslations(locale, ['common'])), diff --git a/src/pages/onedrive-vercel-index-oauth/step-3.tsx b/src/pages/onedrive-vercel-index-oauth/step-3.tsx index 4ddda11c08..cd1a884f95 100755 --- a/src/pages/onedrive-vercel-index-oauth/step-3.tsx +++ b/src/pages/onedrive-vercel-index-oauth/step-3.tsx @@ -12,27 +12,26 @@ import Footer from '../../components/Footer' import { getAuthPersonInfo, requestTokenWithAuthCode, sendTokenToServer } from '../../utils/oAuthHandler' import { LoadingIcon } from '../../components/Loading' -import { getAccessToken } from '../api' // 导入getAccessToken函数 +import { getAccessToken } from '../api' export async function getServerSideProps({ query, locale }) { const { authCode } = query const clientId = process.env.CLIENT_ID || ''; const clientSecret = process.env.CLIENT_SECRET || ''; const userPrincipalName = process.env.USER_PRINCIPAL_NAME || ''; - - // 检查是否已经通过OAuth认证 + + // Check if OAuth authentication has been completed const existingAccessToken = await getAccessToken(); if (existingAccessToken) { - // 如果已经通过OAuth认证,重定向到主页 + // If OAuth authentication has been completed, redirect to the homepage return { - edirect: { - estination: '/', - ermanent: false, + redirect: { + destination: '/', + permanent: false, }, } } - - // 如果没有通过OAuth认证,继续执行OAuth认证的流程 + // If the OAuth authentication is not completed, continue the process of OAuth authentication if (!authCode) { return { props: { diff --git a/src/utils/oAuthHandler.ts b/src/utils/oAuthHandler.ts index 53f8dc5565..056c7d291f 100755 --- a/src/utils/oAuthHandler.ts +++ b/src/utils/oAuthHandler.ts @@ -55,11 +55,7 @@ export function extractAuthCodeFromRedirected(url: string): string { // After a successful authorisation, the code returned from the Microsoft OAuth 2.0 authorization URL // will be used to request an access token. This function requests the access token with the authorisation code // and returns the access token and refresh token on success. -export async function requestTokenWithAuthCode( - code: string, - config: any, - retry = 5 -): Promise< +export async function requestTokenWithAuthCode(code: string, config: any): Promise< | { expiryTime: string; accessToken: string; refreshToken: string } | { error: string; errorDescription: string; errorUri: string } > { @@ -91,26 +87,10 @@ export async function requestTokenWithAuthCode( const { error, error_description, error_uri } = err.response.data return { error, errorDescription: error_description, errorUri: error_uri } }) - } - catch (error) { + } catch (error) { console.error("Failed to get config:", error) - let errorMessage = "" - if (error instanceof Error) { - errorMessage = error.message - } - if (retry > 0) { - console.log(`Retrying... ${retry} attempts left.`) - return requestTokenWithAuthCode(code, retry - 1) - } - else { - if (error instanceof Error) { - return { error: "Failed to get config", errorDescription: error.message, errorUri: "" } - } - else { - // If error is not an instance of Error, we can return a generic error message - return { error: "Failed to get config", errorDescription: "Unknown error", errorUri: "" } - } - } + const errorMessage = error instanceof Error ? error.message : "Unknown error" + return { error: "Failed to get config", errorDescription: errorMessage, errorUri: "" } } } From fea945eb2b0b85d4b58c6a341d7ee59834e6d553 Mon Sep 17 00:00:00 2001 From: eks Date: Thu, 13 Jul 2023 17:31:07 +0800 Subject: [PATCH 88/88] update README --- README.md | 6 +++--- README.zh-CN.md | 4 ++-- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index e277ce18ab..31075fa7e6 100644 --- a/README.md +++ b/README.md @@ -50,13 +50,13 @@ The [Demo](https://odi-demo.freeloop.one) of this One-Click Deploy version. | Th > > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` -4. After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database. +4. **After the initial successful deployment, the deployed page will return a 404 error because we still need to connect to the Redis database.** > `REDIS_URL`:If you are encountering Redis database for the first time, I strongly recommend using Upstash, which is free and deeply integrated with Vercel. For details, refer to [Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration). Follow the instructions to set it up in Vercel's [Upstash Integration](https://vercel.com/integrations/upstash)(simply create a new database in the `Redis` of Upstash, then create a new integration in `Vercel Integrations`, and associate the just deployed OneDrive-Index project with the Redis database), it will automatically fill in the environment variables after project deployment. -- After `REDIS_URL` is successfully set, redeploy the project again. +5. **After `REDIS_URL` is successfully set, redeploy the project again.** -**After successful deployment, when you visit your `onedrive-vercel-index` page for the first time, it will guide you to perform OAuth authentication (quite simple). For details, please refer to the [Instructions](https://ovi.swo.moe/zh/docs/getting-started#authentication) written by the original author.** +6. **After successful deployment, when you visit your `onedrive-vercel-index` page for the first time, it will guide you to perform OAuth authentication (quite simple). For details, please refer to the [Instructions](https://ovi.swo.moe/zh/docs/getting-started#authentication) written by the original author.** ## Environment Variables diff --git a/README.zh-CN.md b/README.zh-CN.md index 5fe82e4503..4d1fdf98c0 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -50,11 +50,11 @@ > > [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/clone?repository-url=https%3A%2F%2Fgithub.com%2FiRedScarf%2Fonedrive-vercel-index&env=NEXT_PUBLIC_SITE_TITLE,USER_PRINCIPAL_NAME,BASE_DIRECTORY,NEXT_PUBLIC_PROTECTED_ROUTES,CLIENT_ID,CLIENT_SECRET,KV_PREFIX) with `NEXT_PUBLIC_PROTECTED_ROUTES` & `KV_PREFIX` -4. 初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。 +4. **初次部署成功后,部署的页面上去是404的,因为我们还需要连接到Redis数据库。** > `REDIS_URL`:如果您是第一次接触Redis数据库,那么强烈推荐您使用免费且与Vercel深度合作的Upstash,详细参考[Vercel Integration](https://docs.upstash.com/redis/howto/vercelintegration),按说明在Vercel的[Upstash集成](https://vercel.com/integrations/upstash)中设置好(简单说就是在Upstash的`Redis`选项卡中新建一个数据库,再在`Vercel Integrations`中新建集成,把刚部署的OneDrive-Index项目与Redis数据库进行关联),它会自动填入项目部署后的环境变量中。 -5. `REDIS_URL`设置成功后,再重新部署一次项目。 +5. **`REDIS_URL`设置成功后,再重新部署一次项目。** 6. **部署成功后,当您第一次访问您的`onedrive-vercel-index`页面时,会引导你进行OAuth认证(相当简单),详情请参考原作者编写的[说明文档](https://ovi.swo.moe/zh/docs/getting-started#进行认证)。**