mirror of
https://github.com/Drop-OSS/drop.git
synced 2025-11-10 04:22:09 +10:00
Compare commits
57 Commits
v0.3.1
...
fa9620eac1
| Author | SHA1 | Date | |
|---|---|---|---|
| fa9620eac1 | |||
| a201b62c04 | |||
| 9bf164ab77 | |||
| 97c6f3490c | |||
| f5cb856d3d | |||
| 67de1f6c02 | |||
| 1002265000 | |||
| 37a2dff0dd | |||
| 799cd6c394 | |||
| 2a005a2222 | |||
| 3942d5c442 | |||
| a520d52ad3 | |||
| aa1de921ee | |||
| bfeacbbdfe | |||
| afce9f159a | |||
| fd828d5b50 | |||
| b33e27e446 | |||
| c97a56eb42 | |||
| 5e5519ece7 | |||
| 6d89b7e510 | |||
| 6baddc10e9 | |||
| a2ea0060cb | |||
| 6aaab30439 | |||
| ea5d108a10 | |||
| f0b127789f | |||
| 4c8be2bfd1 | |||
| 7e371adeb0 | |||
| 6d7b491adb | |||
| abec952e39 | |||
| 9ff541059d | |||
| b84d1f20b5 | |||
| ecc806dc07 | |||
| 45c94cfcbf | |||
| f6f972c2d6 | |||
| e1dc26f676 | |||
| 2fec40c5a6 | |||
| 8f572e1259 | |||
| 43aa15d45c | |||
| 59a5540248 | |||
| 5bfb3e0f68 | |||
| c04f6cbf80 | |||
| d2863fa95b | |||
| 821fd2cf2d | |||
| 6f84ad42fc | |||
| 1d1157a902 | |||
| 6ca9e34c7e | |||
| bc29c468d8 | |||
| 925ea1a414 | |||
| c9addd407e | |||
| 242ae09857 | |||
| ba28c52912 | |||
| a98c95e695 | |||
| 26615ccad0 | |||
| 0b0972b48d | |||
| a435ead916 | |||
| 545a6b154a | |||
| 442f940cc4 |
18
.github/workflows/ci.yml
vendored
18
.github/workflows/ci.yml
vendored
@ -21,17 +21,20 @@ jobs:
|
||||
with:
|
||||
submodules: true
|
||||
|
||||
- name: Install pnpm
|
||||
uses: pnpm/action-setup@v4
|
||||
|
||||
- name: Setup Node.js environment
|
||||
uses: actions/setup-node@v4
|
||||
with:
|
||||
node-version: lts/*
|
||||
cache: "yarn"
|
||||
cache: "pnpm"
|
||||
|
||||
- name: Install dependencies
|
||||
run: yarn install --immutable --network-timeout 1000000
|
||||
run: pnpm install
|
||||
|
||||
- name: Typecheck
|
||||
run: yarn typecheck
|
||||
run: pnpm run typecheck
|
||||
|
||||
lint:
|
||||
name: Lint
|
||||
@ -42,14 +45,17 @@ jobs:
|
||||
with:
|
||||
submodules: true
|
||||
|
||||
- name: Install pnpm
|
||||
uses: pnpm/action-setup@v4
|
||||
|
||||
- name: Setup Node.js environment
|
||||
uses: actions/setup-node@v4
|
||||
with:
|
||||
node-version: lts/*
|
||||
cache: "yarn"
|
||||
cache: "pnpm"
|
||||
|
||||
- name: Install dependencies
|
||||
run: yarn install --immutable --network-timeout 1000000
|
||||
run: pnpm install
|
||||
|
||||
- name: Lint
|
||||
run: yarn lint
|
||||
run: pnpm run lint
|
||||
|
||||
3
.github/workflows/release.yml
vendored
3
.github/workflows/release.yml
vendored
@ -8,9 +8,6 @@ on:
|
||||
schedule:
|
||||
- cron: "0 2 * * *" # run at 2 AM UTC
|
||||
|
||||
permissions:
|
||||
contents: read
|
||||
|
||||
jobs:
|
||||
web:
|
||||
name: Push website Docker image to registry
|
||||
|
||||
@ -1 +1,3 @@
|
||||
drop-base/
|
||||
drop-base/
|
||||
# file is fully managed by pnpm, no reason to break it
|
||||
pnpm-lock.yaml
|
||||
|
||||
243
CONTRIBUTING.md
243
CONTRIBUTING.md
@ -1,242 +1,3 @@
|
||||
# CONTRIBUTING GUIDELINES
|
||||
# Contributing
|
||||
|
||||
Drop is a community-driven project. Contribution is welcome, encouraged, and appreciated.
|
||||
It is also essential for the development of the project.
|
||||
|
||||
First, please take a moment to review our [code of conduct](CODE_OF_CONDUCT.md).
|
||||
|
||||
These guidelines are an attempt at better addressing pending
|
||||
issues and pull requests. Please read them closely.
|
||||
|
||||
Foremost, be so kind as to [search](#use-the-search-luke). This ensures any contribution
|
||||
you would make is not already covered.
|
||||
|
||||
<!-- TOC updateonsave:true depthfrom:2 -->
|
||||
|
||||
- [Reporting Issues](#reporting-issues)
|
||||
- [You have a problem](#you-have-a-problem)
|
||||
- [You have a suggestion](#you-have-a-suggestion)
|
||||
- [Submitting Pull Requests](#submitting-pull-requests)
|
||||
- [Getting started](#getting-started)
|
||||
- [You have a solution](#you-have-a-solution)
|
||||
- [You have an addition](#you-have-an-addition)
|
||||
- [Use the Search, Luke](#use-the-search-luke)
|
||||
- [Commit Guidelines](#commit-guidelines)
|
||||
- [Format](#format)
|
||||
- [Style](#style)
|
||||
|
||||
<!-- /TOC -->
|
||||
|
||||
## Reporting Issues
|
||||
|
||||
### You have a problem
|
||||
|
||||
Please be so kind as to [search](#use-the-search-luke) for any open issue already covering
|
||||
your problem.
|
||||
|
||||
If you find one, comment on it, so we know more people are experiencing it.
|
||||
|
||||
<!--
|
||||
TODO: Add Troubleshooting
|
||||
If not, look at the [Troubleshooting](https://github.com/Drop-OSS/docs/Troubleshooting)
|
||||
page for instructions on how to gather data to better debug your problem.
|
||||
-->
|
||||
|
||||
If you cannot find an existing issue, you can go ahead and create an issue with as much
|
||||
detail as you can provide.
|
||||
It should include the data gathered as indicated above, along with the following:
|
||||
|
||||
1. How to reproduce the problem
|
||||
2. What the correct behavior should be
|
||||
3. What the actual behavior is
|
||||
|
||||
Please copy to anyone relevant (e.g. plugin maintainers) by mentioning their GitHub handle
|
||||
(starting with `@`) in your message.
|
||||
|
||||
We will do our very best to help you.
|
||||
|
||||
### You have a suggestion
|
||||
|
||||
Please be so kind as to [search](#use-the-search-luke) for any open issue already covering
|
||||
your suggestion.
|
||||
|
||||
If you find one, comment on it, so we know more people are supporting it.
|
||||
|
||||
If not, you can go ahead and create an issue. Please copy to anyone relevant (e.g. plugin
|
||||
maintainers) by mentioning their GitHub handle (starting with `@`) in your message.
|
||||
|
||||
## Submitting Pull Requests
|
||||
|
||||
### Getting started
|
||||
|
||||
You should be familiar with the basics of
|
||||
[contributing on GitHub](https://help.github.com/articles/using-pull-requests)
|
||||
|
||||
<!--and have a fork
|
||||
[properly set up](https://github.com/drop/docs/Contribution-Technical-Practices).
|
||||
-->
|
||||
|
||||
You MUST always create PRs with _a dedicated branch_ based on the latest upstream tree.
|
||||
|
||||
If you create your own PR, please make sure you do it right. Also be so kind as to reference
|
||||
any issue that would be solved in the PR description body,
|
||||
[for instance](https://help.github.com/articles/closing-issues-via-commit-messages/)
|
||||
_"Fixes #XXXX"_ for issue number XXXX.
|
||||
|
||||
### You have a solution
|
||||
|
||||
Please be so kind as to [search](#use-the-search-luke) for any open issue already covering
|
||||
your [problem](#you-have-a-problem), and any pending/merged/rejected PR covering your solution.
|
||||
|
||||
If the solution is already reported, try it out and +1 the pull request if the
|
||||
solution works ok. On the other hand, if you think your solution is better, post
|
||||
it with reference to the other one so we can have both solutions to compare.
|
||||
|
||||
If not, then go ahead and submit a PR. Please copy to anyone relevant (e.g. plugin
|
||||
maintainers) by mentioning their GitHub handle (starting with `@`) in your message.
|
||||
|
||||
### You have an addition
|
||||
|
||||
We are absolutely accepting more contributions or features to drop, but please, make sure
|
||||
that it is reasonable. Contributions that only cover a very small niche are likely to not
|
||||
be added.
|
||||
|
||||
Please be so kind as to [search](#use-the-search-luke) for any pending, merged or rejected Pull Requests
|
||||
covering or related to what you want to add.
|
||||
|
||||
If you find one, try it out and work with the author on a common solution.
|
||||
|
||||
If not, then go ahead and submit a PR. Please copy to anyone relevant (e.g. plugin
|
||||
maintainers) by mentioning their GitHub handle (starting with `@`) in your message.
|
||||
|
||||
For any extensive change, such as API changes, you will have to find testers to +1 your PR.
|
||||
|
||||
---
|
||||
|
||||
## Use the Search, Luke
|
||||
|
||||
_May the Force (of past experiences) be with you_
|
||||
|
||||
GitHub offers [many search features](https://help.github.com/articles/searching-github/)
|
||||
to help you check whether a similar contribution to yours already exists. Please search
|
||||
before making any contribution, it avoids duplicates and eases maintenance. Trust me,
|
||||
that works 90% of the time.
|
||||
|
||||
You can also take a look at the [FAQ](https://github.com/Drop-OSS/docs/wiki/FAQ)
|
||||
to be sure your contribution has not already come up.
|
||||
|
||||
If all fails, your thing has probably not been reported yet, so you can go ahead
|
||||
and [create an issue](#reporting-issues) or [submit a PR](#submitting-pull-requests).
|
||||
|
||||
---
|
||||
|
||||
## Translation
|
||||
|
||||
If you want to help translate Drop, we would love to have your help! You can do so on our weblate instance. Please make sure to read the [message format syntax](https://vue-i18n.intlify.dev/guide/essentials/syntax.html) page before starting. Failure to do so may result in your translations causing errors in Drop.
|
||||
|
||||
## Commit Guidelines
|
||||
|
||||
Drop uses the [Conventional Commits](https://www.conventionalcommits.org/en/v1.0.0/)
|
||||
specification. The automatic changelog tool uses these to automatically generate
|
||||
a changelog based on the commit messages. Here's a guide to writing a commit message
|
||||
to allow this:
|
||||
|
||||
### Format
|
||||
|
||||
```
|
||||
type(scope)!: subject
|
||||
```
|
||||
|
||||
- `type`: the type of the commit is one of the following:
|
||||
- `feat`: new features.
|
||||
- `fix`: bug fixes.
|
||||
- `docs`: documentation changes.
|
||||
- `refactor`: refactor of a particular code section without introducing
|
||||
new features or bug fixes.
|
||||
- `style`: code style improvements.
|
||||
- `perf`: performance improvements.
|
||||
- `test`: changes to the test suite.
|
||||
- `ci`: changes to the CI system.
|
||||
- `build`: changes to the build system.
|
||||
- `chore`: for other changes that don't match previous types. This doesn't appear
|
||||
in the changelog.
|
||||
|
||||
- `scope`: section of the codebase that the commit makes changes to. If it makes changes to
|
||||
many sections, or if no section in particular is modified, leave blank without the parentheses.
|
||||
Examples:
|
||||
- Commit that changes the `git` plugin:
|
||||
|
||||
```
|
||||
feat(git): add alias for `git commit`
|
||||
```
|
||||
|
||||
- Commit that changes many plugins:
|
||||
|
||||
```
|
||||
style: fix inline declaration of arrays
|
||||
```
|
||||
|
||||
For changes to plugins or themes, the scope should be the plugin or theme name:
|
||||
- ✅ `fix(agnoster): commit subject`
|
||||
- ❌ `fix(theme/agnoster): commit subject`
|
||||
|
||||
- `!`: this goes after the `scope` (or the `type` if scope is empty), to indicate that the commit
|
||||
introduces breaking changes.
|
||||
|
||||
Optionally, you can specify a message that the changelog tool will display to the user to indicate
|
||||
what's changed and what they can do to deal with it. You can use multiple lines to type this message;
|
||||
the changelog parser will keep reading until the end of the commit message or until it finds an empty
|
||||
line.
|
||||
|
||||
Example (made up):
|
||||
|
||||
```
|
||||
style(agnoster)!: change dirty git repo glyph
|
||||
|
||||
BREAKING CHANGE: the glyph to indicate when a git repository is dirty has
|
||||
changed from a Powerline character to a standard UTF-8 emoji. You can
|
||||
change it back by setting `ZSH_THEME_DIRTY_GLYPH`.
|
||||
|
||||
Fixes #420
|
||||
|
||||
Co-authored-by: Username <email>
|
||||
```
|
||||
|
||||
- `subject`: a brief description of the changes. This will be displayed in the changelog. If you need
|
||||
to specify other details, you can use the commit body, but it won't be visible.
|
||||
|
||||
Formatting tricks: the commit subject may contain:
|
||||
- Links to related issues or PRs by writing `#issue`. This will be highlighted by the changelog tool:
|
||||
|
||||
```
|
||||
feat(archlinux): add support for aura AUR helper (#9467)
|
||||
```
|
||||
|
||||
- Formatted inline code by using backticks: the text between backticks will also be highlighted by
|
||||
the changelog tool:
|
||||
```
|
||||
feat(shell-proxy): enable unexported `DEFAULT_PROXY` setting (#9774)
|
||||
```
|
||||
|
||||
### Style
|
||||
|
||||
Try to keep the first commit line short. It's harder to do using this commit style but try to be
|
||||
concise, and if you need more space, you can use the commit body. Try to make sure that the commit
|
||||
subject is clear and precise enough that users will know what changed by just looking at the changelog.
|
||||
|
||||
---
|
||||
|
||||
<!--
|
||||
## Volunteer
|
||||
|
||||
Very nice!! :)
|
||||
|
||||
Please have a look at the [Volunteer](https://github.com/ohmyzsh/ohmyzsh/wiki/Volunteers)
|
||||
page for instructions on where to start and more.
|
||||
-->
|
||||
|
||||
## Reference
|
||||
|
||||
This contributing guide is adapted from the
|
||||
[oh-my-zsh contribution guide](https://github.com/ohmyzsh/ohmyzsh/blob/master/CONTRIBUTING.md).
|
||||
If there are any issues with this, please email admin@deepcore.dev.
|
||||
Check out our contributing guidelines on our developer docs: [https://developer.droposs.org/contributing](https://developer.droposs.org/contributing).
|
||||
|
||||
41
Dockerfile
41
Dockerfile
@ -1,40 +1,45 @@
|
||||
# syntax=docker/dockerfile:1
|
||||
|
||||
FROM node:lts-alpine AS base
|
||||
ENV PNPM_HOME="/pnpm"
|
||||
ENV PATH="$PNPM_HOME:$PATH"
|
||||
RUN corepack enable
|
||||
WORKDIR /app
|
||||
|
||||
# so corepack knows pnpm's version
|
||||
COPY package.json pnpm-lock.yaml pnpm-workspace.yaml ./
|
||||
# prevent prompt to download
|
||||
ENV COREPACK_ENABLE_DOWNLOAD_PROMPT=0
|
||||
# setup for offline
|
||||
RUN corepack pack
|
||||
# don't call out to network anymore
|
||||
ENV COREPACK_ENABLE_NETWORK=0
|
||||
|
||||
### Unified deps builder
|
||||
# FROM node:lts-alpine AS deps
|
||||
# WORKDIR /app
|
||||
# COPY package.json yarn.lock ./
|
||||
# RUN --mount=type=cache,target=/root/.yarn YARN_CACHE_FOLDER=/root/.yarn yarn install --network-timeout 1000000 --ignore-scripts
|
||||
FROM base AS deps
|
||||
RUN pnpm install --frozen-lockfile --ignore-scripts
|
||||
|
||||
### Build for app
|
||||
FROM node:lts-alpine AS build-system
|
||||
# setup workdir - has to be the same filepath as app because fuckin' Prisma
|
||||
WORKDIR /app
|
||||
FROM base AS build-system
|
||||
|
||||
ENV NODE_ENV=production
|
||||
ENV NUXT_TELEMETRY_DISABLED=1
|
||||
# ENV YARN_CACHE_FOLDER=/root/.yarn
|
||||
|
||||
# add git so drop can determine its git ref at build
|
||||
# pnpm for build
|
||||
RUN apk add --no-cache git pnpm
|
||||
RUN apk add --no-cache git
|
||||
|
||||
# copy deps and rest of project files
|
||||
# COPY --from=deps /app/node_modules ./node_modules
|
||||
COPY --from=deps /app/node_modules ./node_modules
|
||||
COPY . .
|
||||
|
||||
ARG BUILD_DROP_VERSION
|
||||
ARG BUILD_GIT_REF
|
||||
|
||||
# build
|
||||
RUN pnpm import
|
||||
RUN pnpm install --shamefully-hoist
|
||||
RUN pnpm run build
|
||||
# RUN --mount=type=cache,target=/root/.yarn yarn postinstall && yarn build
|
||||
RUN pnpm run postinstall && pnpm run build
|
||||
|
||||
### create run environment for Drop
|
||||
FROM node:lts-alpine AS run-system
|
||||
WORKDIR /app
|
||||
FROM base AS run-system
|
||||
|
||||
ENV NODE_ENV=production
|
||||
ENV NUXT_TELEMETRY_DISABLED=1
|
||||
@ -42,6 +47,8 @@ ENV NUXT_TELEMETRY_DISABLED=1
|
||||
# RUN --mount=type=cache,target=/root/.yarn YARN_CACHE_FOLDER=/root/.yarn yarn add --network-timeout 1000000 --no-lockfile --ignore-scripts prisma@6.11.1
|
||||
RUN apk add --no-cache pnpm
|
||||
RUN pnpm install prisma@6.11.1
|
||||
# init prisma to download all required files
|
||||
RUN pnpm prisma init
|
||||
|
||||
COPY --from=build-system /app/package.json ./
|
||||
COPY --from=build-system /app/.output ./app
|
||||
|
||||
71
README.md
71
README.md
@ -6,73 +6,32 @@
|
||||
# Drop
|
||||
|
||||
[](https://droposs.org)
|
||||
[](https://docs.droposs.org/)
|
||||
[](https://forum.droposs.org)
|
||||
[](LICENSE)
|
||||
[](https://discord.gg/ACq4qZp4a9)
|
||||
[](https://opencollective.com/drop-oss)
|
||||
[
|
||||
](https://translate.droposs.org/engage/drop/)
|
||||
|
||||
Drop is an open-source game distribution platform, like GameVault or Steam. It's designed to distribute and shared DRM-free game quickly, all while being incredibly flexible, beautiful and fast.
|
||||
Drop is an open-source game distribution platform, similar to GameVault or Steam. It's designed to distribute and share DRM-free games quickly, all while being incredibly flexible, beautiful, and fast.
|
||||
|
||||
<div align="center">
|
||||
<img src="https://droposs.org/_ipx/f_webp&q_80/images/carousel/store.png" alt="Drop Screenshot" width="900rem"/>
|
||||
</div>
|
||||
|
||||
## Philosophy
|
||||
|
||||
1. Drop is flexible. While abstractions and interfaces can make the codebase more complicated, the flexibility is worth it.
|
||||
2. Drop is secure. The nature of Drop means an instance can never be accessible without authentication. In line with #1, Drop also supports a huge variety of authentication mechanisms, from a username/password to SSO.
|
||||
3. Drop is user-friendly. The interface is designed to be clean and simple to use, with complexity available to the users who want it.
|
||||
1. Drop is flexible. While abstractions and interfaces can complicate the codebase, the flexibility is worth it.
|
||||
2. Drop is secure. The nature of Drop means an instance can never be accessible without authentication. In line with #1, Drop also supports a huge variety of authentication mechanisms, from username/password to SSO.
|
||||
3. Drop is user-friendly. The interface is designed to be clean and simple to use, with advanced features available to users who want them.
|
||||
|
||||
## Deployment
|
||||
|
||||
To just deploy Drop, we've set up a simple docker compose file in deploy-template.
|
||||
|
||||
1. Generate a [GiantBomb API Key](https://www.giantbomb.com/api/)
|
||||
2. Navigate to the deploy-template directory in your terminal (`cd deploy-template`)
|
||||
3. Edit the compose.yml file (`nano compose.yml`) and copy your GiamtBomb API Key into the GIANT_BOMB_API_KEY environment variable
|
||||
4. Run `docker compose up -d`
|
||||
|
||||
Your drop server should now be running. To register the admin user, navigate to http://your.drop.server.ip:3000/register?id=admin
|
||||
and fill in the required forms
|
||||
|
||||
### Adding a game
|
||||
|
||||
To add a game to the drop library, do as follows:
|
||||
|
||||
1. Ensure that the current user owns the library folder with `sudo chown -R $(id -u $(whoami)) library`
|
||||
2. `cd library`
|
||||
3. `mkdir <GAME_NAME>` with the name of the game which you would like to register
|
||||
4. `cd <GAME_NAME>`
|
||||
5. `mkdir <VERSION_NAME>` Upload files for the specific game version to this folder
|
||||
6. Navigate to http://your.drop.server.ip:3000/
|
||||
7. Import game metadata (uses GiantBomb API Key) by selecting the game and specifying which entry to import
|
||||
8. Navigate to http://your.drop.server.ip:3000/admin/library
|
||||
9. You should see the game which you have just imported listed in this menu. There should be a notification that "Drop has detected you have new verions of this game to import". Select import here.
|
||||
10. Select the game version to import and thus fill in fields as required.
|
||||
|
||||
## Tech Stack
|
||||
|
||||
This repo uses the Nuxt 3 + TailwindCSS stack, with the `yarn` package manager.
|
||||
|
||||
For the database, Drop uses Prisma connected to PostgreSQL.
|
||||
|
||||
## Development
|
||||
|
||||
To get started with development, you need `yarn --optional` and `docker compose` installed (or know how to set up a PostgreSQL database).
|
||||
|
||||
### Note: `--optional` flag is **REQUIRED**
|
||||
|
||||
Drop uses a utility package called droplet that's written in Rust. It has builts for Linux (GNU) and Windows, and they are set up as optional packages. `npm` installs these by default, but `yarn` needs the `--optional` flag.
|
||||
|
||||
Steps:
|
||||
|
||||
1. Run `git submodule update --init --recursive` to setup submodules
|
||||
1. Copy the `.env.example` to `.env` and add your GiantBomb metadata key (more metadata providers coming)
|
||||
1. Create the `.data` directory with `mkdir .data`
|
||||
1. Ensure that your user owns the `.data` directory with `sudo chown -R $(id -u $(whoami))`
|
||||
1. Open up a terminal and navigate to `dev-tools`, and run `docker compose up`
|
||||
1. Open up another terminal in the root directory of the project and run `yarn` and then `yarn dev` to start the dev server
|
||||
|
||||
As part of the first-time bootstrap, Drop creates an invitation with the fixed id of 'admin'. So, to create an admin account, go to:
|
||||
|
||||
http://localhost:3000/auth/register?id=admin
|
||||
See our documentation on how to [deploy Drop](https://docs.droposs.org/docs/guides/quickstart) for more information.
|
||||
|
||||
## Contributing
|
||||
|
||||
Please see the [in-depth contributing guide](CONTRIBUTING.md)
|
||||
Please see the [in-depth contributing guide](CONTRIBUTING.md). The guide includes information on how to set up the project, how to contribute code, how to report issues, and even how to effectively translate Drop.
|
||||
|
||||
[](https://translate.droposs.org/engage/drop/)
|
||||
|
||||
42
app.vue
42
app.vue
@ -4,10 +4,52 @@
|
||||
<NuxtPage />
|
||||
</NuxtLayout>
|
||||
<ModalStack />
|
||||
<div
|
||||
v-if="showExternalUrlWarning"
|
||||
class="fixed flex flex-row gap-x-2 right-0 bottom-0 m-2 px-2 py-2 z-50 text-right bg-red-700/90 rounded-lg"
|
||||
>
|
||||
<div class="flex flex-col">
|
||||
<span class="text-sm text-zinc-200 font-bold font-display">{{
|
||||
$t("errors.externalUrl.title")
|
||||
}}</span>
|
||||
<span class="text-xs text-red-400">{{
|
||||
$t("errors.externalUrl.subtitle")
|
||||
}}</span>
|
||||
</div>
|
||||
<button class="text-red-200" @click="() => hideExternalURL()">
|
||||
<XMarkIcon class="size-5" />
|
||||
</button>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { XMarkIcon } from "@heroicons/vue/24/outline";
|
||||
|
||||
await updateUser();
|
||||
|
||||
const user = useUser();
|
||||
const apiDetails = await $dropFetch("/api/v1");
|
||||
|
||||
const showExternalUrlWarning = ref(false);
|
||||
function checkExternalUrl() {
|
||||
if (!import.meta.client) return;
|
||||
const realOrigin = window.location.origin.trim();
|
||||
const chosenOrigin = apiDetails.external.trim();
|
||||
const ignore = window.localStorage.getItem("ignoreExternalUrl");
|
||||
if (ignore && ignore == "true") return;
|
||||
showExternalUrlWarning.value = !(realOrigin == chosenOrigin);
|
||||
}
|
||||
|
||||
function hideExternalURL() {
|
||||
window.localStorage.setItem("ignoreExternalUrl", "true");
|
||||
showExternalUrlWarning.value = false;
|
||||
}
|
||||
|
||||
if (user.value?.admin) {
|
||||
onMounted(() => {
|
||||
checkExternalUrl();
|
||||
});
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
@ -45,6 +45,7 @@ import {
|
||||
LockClosedIcon,
|
||||
DevicePhoneMobileIcon,
|
||||
WrenchScrewdriverIcon,
|
||||
CodeBracketIcon,
|
||||
} from "@heroicons/vue/24/outline";
|
||||
import { UserIcon } from "@heroicons/vue/24/solid";
|
||||
import type { Component } from "vue";
|
||||
@ -73,6 +74,12 @@ const navigation: (NavigationItem & { icon: Component; count?: number })[] = [
|
||||
icon: BellIcon,
|
||||
count: notifications.value.length,
|
||||
},
|
||||
{
|
||||
label: t("account.token.title"),
|
||||
route: "/account/tokens",
|
||||
prefix: "/account/tokens",
|
||||
icon: CodeBracketIcon,
|
||||
},
|
||||
{
|
||||
label: t("account.settings"),
|
||||
route: "/account/settings",
|
||||
|
||||
@ -4,9 +4,10 @@
|
||||
v-for="(_, i) in amount"
|
||||
:key="i"
|
||||
:class="[
|
||||
carousel.currentSlide == i ? 'bg-blue-600 w-6' : 'bg-zinc-700 w-3',
|
||||
carousel.currentSlide === i ? 'bg-blue-600 w-6' : 'bg-zinc-700 w-3',
|
||||
'transition-all cursor-pointer h-2 rounded-full',
|
||||
]"
|
||||
@click="slideTo(i)"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
@ -18,8 +19,8 @@ const carousel = inject(injectCarousel)!;
|
||||
|
||||
const amount = carousel.maxSlide - carousel.minSlide + 1;
|
||||
|
||||
// function slideTo(index: number) {
|
||||
// const offsetIndex = index + carousel.minSlide;
|
||||
// carousel.nav.slideTo(offsetIndex);
|
||||
// }
|
||||
function slideTo(index: number) {
|
||||
const offsetIndex = index + carousel.minSlide;
|
||||
carousel.nav.slideTo(offsetIndex);
|
||||
}
|
||||
</script>
|
||||
|
||||
@ -1,5 +1,6 @@
|
||||
<template>
|
||||
<svg
|
||||
aria-label="Drop Logo"
|
||||
class="text-blue-400"
|
||||
viewBox="0 0 24 24"
|
||||
fill="none"
|
||||
@ -9,6 +10,16 @@
|
||||
d="M4 13.5C4 11.0008 5.38798 8.76189 7.00766 7C8.43926 5.44272 10.0519 4.25811 11.0471 3.5959C11.6287 3.20893 12.3713 3.20893 12.9529 3.5959C13.9481 4.25811 15.5607 5.44272 16.9923 7C18.612 8.76189 20 11.0008 20 13.5C20 17.9183 16.4183 21.5 12 21.5C7.58172 21.5 4 17.9183 4 13.5Z"
|
||||
stroke="currentColor"
|
||||
stroke-width="2"
|
||||
stroke-dasharray="100"
|
||||
:stroke-dashoffset="dashArray"
|
||||
/>
|
||||
</svg>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{ progress?: number }>();
|
||||
|
||||
const dashArray = computed(() =>
|
||||
props.progress === undefined ? 0 : ((100 - props.progress) / 100) * 50 + 50,
|
||||
);
|
||||
</script>
|
||||
|
||||
@ -10,7 +10,7 @@
|
||||
d="M203.371.916c-26.013-2.078-76.686 1.963-124.73 9.946L67.3 12.749C35.421 18.062 18.2 21.766 6.004 25.934 1.244 27.561.828 27.778.874 28.61c.07 1.214.828 1.121 9.595-1.176 9.072-2.377 17.15-3.92 39.246-7.496C123.565 7.986 157.869 4.492 195.942 5.046c7.461.108 19.25 1.696 19.17 2.582-.107 1.183-7.874 4.31-25.75 10.366-21.992 7.45-35.43 12.534-36.701 13.884-2.173 2.308-.202 4.407 4.442 4.734 2.654.187 3.263.157 15.593-.78 35.401-2.686 57.944-3.488 88.365-3.143 46.327.526 75.721 2.23 130.788 7.584 19.787 1.924 20.814 1.98 24.557 1.332l.066-.011c1.201-.203 1.53-1.825.399-2.335-2.911-1.31-4.893-1.604-22.048-3.261-57.509-5.556-87.871-7.36-132.059-7.842-23.239-.254-33.617-.116-50.627.674-11.629.54-42.371 2.494-46.696 2.967-2.359.259 8.133-3.625 26.504-9.81 23.239-7.825 27.934-10.149 28.304-14.005.417-4.348-3.529-6-16.878-7.066Z"
|
||||
/>
|
||||
</svg>
|
||||
<DropLogo class="h-6" />
|
||||
<DropLogo aria-hidden="true" class="h-6" />
|
||||
<span class="text-blue-400 font-display font-bold text-xl uppercase">
|
||||
{{ $t("drop.drop") }}
|
||||
</span>
|
||||
|
||||
@ -22,21 +22,17 @@
|
||||
<!-- import games button -->
|
||||
|
||||
<NuxtLink
|
||||
:href="
|
||||
unimportedVersions.length > 0
|
||||
? `/admin/library/${game.id}/import`
|
||||
: ''
|
||||
"
|
||||
:href="canImport ? `/admin/library/${game.id}/import` : ''"
|
||||
type="button"
|
||||
:class="[
|
||||
unimportedVersions.length > 0
|
||||
canImport
|
||||
? 'bg-blue-600 hover:bg-blue-700'
|
||||
: 'bg-blue-800/50',
|
||||
'inline-flex w-fit items-center gap-x-2 rounded-md px-3 py-1 text-sm font-semibold font-display text-white shadow-sm focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-blue-600',
|
||||
]"
|
||||
>
|
||||
{{
|
||||
unimportedVersions.length > 0
|
||||
canImport
|
||||
? $t("library.admin.import.version.import")
|
||||
: $t("library.admin.import.version.noVersions")
|
||||
}}
|
||||
@ -124,10 +120,16 @@ import { ExclamationCircleIcon } from "@heroicons/vue/24/outline";
|
||||
|
||||
// TODO implement UI for this page
|
||||
|
||||
defineProps<{ unimportedVersions: string[] }>();
|
||||
const props = defineProps<{ unimportedVersions: string[] }>();
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const hasDeleted = ref(false);
|
||||
|
||||
const canImport = computed(
|
||||
() => hasDeleted.value || props.unimportedVersions.length > 0,
|
||||
);
|
||||
|
||||
type GameAndVersions = GameModel & { versions: GameVersionModel[] };
|
||||
const game = defineModel<SerializeObject<GameAndVersions>>() as Ref<
|
||||
SerializeObject<GameAndVersions>
|
||||
@ -176,6 +178,7 @@ async function deleteVersion(versionName: string) {
|
||||
game.value.versions.findIndex((e) => e.versionName === versionName),
|
||||
1,
|
||||
);
|
||||
hasDeleted.value = true;
|
||||
} catch (e) {
|
||||
createModal(
|
||||
ModalType.Notification,
|
||||
|
||||
@ -3,7 +3,7 @@
|
||||
<LanguageSelectorListbox />
|
||||
<NuxtLink
|
||||
class="mt-1 transition text-blue-500 hover:text-blue-600 text-sm"
|
||||
to="https://translate.droposs.org/projects/drop/"
|
||||
to="https://translate.droposs.org/engage/drop/"
|
||||
target="_blank"
|
||||
>
|
||||
<i18n-t
|
||||
@ -18,8 +18,12 @@
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
|
||||
<DevOnly
|
||||
><h1 class="mt-3 text-sm text-gray-500">{{ $t("welcome") }}</h1>
|
||||
<DevOnly>
|
||||
<h1 class="mt-3 text-sm text-gray-500">{{ $t("welcome") }}</h1>
|
||||
</DevOnly>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { ArrowTopRightOnSquareIcon } from "@heroicons/vue/24/outline";
|
||||
</script>
|
||||
|
||||
27
components/LogLine.vue
Normal file
27
components/LogLine.vue
Normal file
@ -0,0 +1,27 @@
|
||||
<template>
|
||||
<span class="text-xs font-mono text-zinc-400 inline-flex items-top gap-x-2"
|
||||
><span v-if="!short" class="text-zinc-500">{{ log.timestamp }}</span>
|
||||
<span
|
||||
:class="[
|
||||
colours[log.level] || 'text-green-400',
|
||||
'uppercase font-display font-semibold',
|
||||
]"
|
||||
>{{ log.level }}</span
|
||||
>
|
||||
<pre :class="[short ? 'line-clamp-1' : '', 'mt-[1px]']">{{
|
||||
log.message
|
||||
}}</pre>
|
||||
</span>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import type { TaskLog } from "~/server/internal/tasks";
|
||||
|
||||
defineProps<{ log: typeof TaskLog.infer; short?: boolean }>();
|
||||
|
||||
const colours: { [key: string]: string } = {
|
||||
info: "text-blue-400",
|
||||
warn: "text-yellow-400",
|
||||
error: "text-red-400",
|
||||
};
|
||||
</script>
|
||||
148
components/Modal/CreateCompany.vue
Normal file
148
components/Modal/CreateCompany.vue
Normal file
@ -0,0 +1,148 @@
|
||||
<template>
|
||||
<ModalTemplate v-model="open">
|
||||
<template #default>
|
||||
<div>
|
||||
<h3 class="text-lg font-medium leading-6 text-white">
|
||||
{{ $t("library.admin.metadata.companies.modals.createTitle") }}
|
||||
</h3>
|
||||
<p class="mt-1 text-zinc-400 text-sm">
|
||||
{{ $t("library.admin.metadata.companies.modals.createDescription") }}
|
||||
</p>
|
||||
</div>
|
||||
<div class="mt-2">
|
||||
<form class="space-y-4" @submit.prevent="() => createCompany()">
|
||||
<div>
|
||||
<label
|
||||
for="name"
|
||||
class="block text-sm/6 font-medium text-zinc-100"
|
||||
>{{
|
||||
$t("library.admin.metadata.companies.modals.createFieldName")
|
||||
}}</label
|
||||
>
|
||||
<div class="mt-2">
|
||||
<input
|
||||
id="name"
|
||||
v-model="companyName"
|
||||
type="text"
|
||||
name="name"
|
||||
:placeholder="
|
||||
$t(
|
||||
'library.admin.metadata.companies.modals.createFieldNamePlaceholder',
|
||||
)
|
||||
"
|
||||
class="block w-full rounded-md bg-zinc-800 px-3 py-1.5 text-base text-zinc-100 outline outline-1 -outline-offset-1 outline-zinc-700 placeholder:text-zinc-400 focus:outline focus:outline-2 focus:-outline-offset-2 focus:outline-blue-600 sm:text-sm/6"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
<label
|
||||
for="description"
|
||||
class="block text-sm/6 font-medium text-zinc-100"
|
||||
>{{
|
||||
$t(
|
||||
"library.admin.metadata.companies.modals.createFieldDescription",
|
||||
)
|
||||
}}</label
|
||||
>
|
||||
<div class="mt-2">
|
||||
<input
|
||||
id="description"
|
||||
v-model="companyDescription"
|
||||
type="text"
|
||||
name="description"
|
||||
:placeholder="
|
||||
$t(
|
||||
'library.admin.metadata.companies.modals.createFieldDescriptionPlaceholder',
|
||||
)
|
||||
"
|
||||
class="block w-full rounded-md bg-zinc-800 px-3 py-1.5 text-base text-zinc-100 outline outline-1 -outline-offset-1 outline-zinc-700 placeholder:text-zinc-400 focus:outline focus:outline-2 focus:-outline-offset-2 focus:outline-blue-600 sm:text-sm/6"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
<label
|
||||
for="website"
|
||||
class="block text-sm/6 font-medium text-zinc-100"
|
||||
>{{
|
||||
$t("library.admin.metadata.companies.modals.createFieldWebsite")
|
||||
}}</label
|
||||
>
|
||||
<div class="mt-2">
|
||||
<input
|
||||
id="website"
|
||||
v-model="companyWebsite"
|
||||
type="text"
|
||||
name="website"
|
||||
:placeholder="
|
||||
$t(
|
||||
'library.admin.metadata.companies.modals.createFieldWebsitePlaceholder',
|
||||
)
|
||||
"
|
||||
class="block w-full rounded-md bg-zinc-800 px-3 py-1.5 text-base text-zinc-100 outline outline-1 -outline-offset-1 outline-zinc-700 placeholder:text-zinc-400 focus:outline focus:outline-2 focus:-outline-offset-2 focus:outline-blue-600 sm:text-sm/6"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<button class="hidden" type="submit" />
|
||||
</form>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<template #buttons="{ close }">
|
||||
<LoadingButton
|
||||
:loading="loading"
|
||||
:disabled="!companyValid"
|
||||
class="w-full sm:w-fit"
|
||||
@click="() => createCompany()"
|
||||
>
|
||||
{{ $t("common.create") }}
|
||||
</LoadingButton>
|
||||
<button
|
||||
ref="cancelButtonRef"
|
||||
type="button"
|
||||
class="mt-3 inline-flex w-full justify-center rounded-md bg-zinc-800 px-3 py-2 text-sm font-semibold text-zinc-100 shadow-sm ring-1 ring-inset ring-zinc-800 hover:bg-zinc-900 sm:mt-0 sm:w-auto"
|
||||
@click="() => close()"
|
||||
>
|
||||
{{ $t("cancel") }}
|
||||
</button>
|
||||
</template>
|
||||
</ModalTemplate>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import type { CompanyModel } from "~/prisma/client/models";
|
||||
|
||||
const open = defineModel<boolean>({ required: true });
|
||||
|
||||
const emit = defineEmits<{
|
||||
created: [company: CompanyModel];
|
||||
}>();
|
||||
|
||||
const companyName = ref("");
|
||||
const companyDescription = ref("");
|
||||
const companyWebsite = ref("");
|
||||
|
||||
const loading = ref(false);
|
||||
const companyValid = computed(
|
||||
() => companyName.value && companyDescription.value,
|
||||
);
|
||||
async function createCompany() {
|
||||
loading.value = true;
|
||||
try {
|
||||
const newCompany = await $dropFetch("/api/v1/admin/company", {
|
||||
method: "POST",
|
||||
body: {
|
||||
name: companyName.value,
|
||||
description: companyDescription.value,
|
||||
website: companyWebsite.value,
|
||||
},
|
||||
failTitle: "Failed to create new company",
|
||||
});
|
||||
open.value = false;
|
||||
emit("created", newCompany);
|
||||
} finally {
|
||||
/* empty */
|
||||
}
|
||||
loading.value = false;
|
||||
}
|
||||
</script>
|
||||
267
components/Modal/CreateToken.vue
Normal file
267
components/Modal/CreateToken.vue
Normal file
@ -0,0 +1,267 @@
|
||||
<template>
|
||||
<ModalTemplate v-model="model" size-class="max-w-3xl">
|
||||
<template #default>
|
||||
<div class="space-y-5">
|
||||
<div>
|
||||
<label
|
||||
for="name"
|
||||
class="block text-sm font-medium leading-6 text-zinc-100"
|
||||
>{{ $t("account.token.name") }}</label
|
||||
>
|
||||
<p class="text-zinc-400 block text-xs font-medium leading-6">
|
||||
{{ $t("account.token.nameDesc") }}
|
||||
</p>
|
||||
<div class="mt-2">
|
||||
<input
|
||||
id="name"
|
||||
v-model="name"
|
||||
name="name"
|
||||
type="text"
|
||||
autocomplete="name"
|
||||
:placeholder="
|
||||
props.suggestedName ?? $t('account.token.namePlaceholder')
|
||||
"
|
||||
class="block w-full rounded-md border-0 py-1.5 px-3 bg-zinc-800 disabled:bg-zinc-900/80 text-zinc-100 disabled:text-zinc-400 shadow-sm ring-1 ring-inset ring-zinc-700 disabled:ring-zinc-800 placeholder:text-zinc-400 focus:ring-2 focus:ring-inset focus:ring-blue-600 sm:text-sm sm:leading-6"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<Listbox v-model="expiryKey" as="div">
|
||||
<ListboxLabel class="block text-sm/6 font-medium text-zinc-100">{{
|
||||
$t("users.admin.simple.inviteExpiryLabel")
|
||||
}}</ListboxLabel>
|
||||
<div class="relative mt-2">
|
||||
<ListboxButton
|
||||
class="relative w-full cursor-default rounded-md bg-zinc-800 py-1.5 pl-3 pr-10 text-left text-zinc-100 shadow-sm ring-1 ring-inset ring-zinc-700 focus:outline-none focus:ring-2 focus:ring-blue-600 sm:text-sm/6"
|
||||
>
|
||||
<span class="block truncate">{{ expiryKey }}</span>
|
||||
<span
|
||||
class="pointer-events-none absolute inset-y-0 right-0 flex items-center pr-2"
|
||||
>
|
||||
<ChevronUpDownIcon
|
||||
class="h-5 w-5 text-gray-400"
|
||||
aria-hidden="true"
|
||||
/>
|
||||
</span>
|
||||
</ListboxButton>
|
||||
|
||||
<transition
|
||||
leave-active-class="transition ease-in duration-100"
|
||||
leave-from-class="opacity-100"
|
||||
leave-to-class="opacity-0"
|
||||
>
|
||||
<ListboxOptions
|
||||
class="absolute z-10 mt-1 max-h-60 w-full overflow-auto rounded-md bg-zinc-900 py-1 text-base shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none sm:text-sm"
|
||||
>
|
||||
<ListboxOption
|
||||
v-for="[label] in Object.entries(expiry)"
|
||||
:key="label"
|
||||
v-slot="{ active, selected }"
|
||||
as="template"
|
||||
:value="label"
|
||||
>
|
||||
<li
|
||||
:class="[
|
||||
active ? 'bg-blue-600 text-white' : 'text-zinc-300',
|
||||
'relative cursor-default select-none py-2 pl-3 pr-9',
|
||||
]"
|
||||
>
|
||||
<span
|
||||
:class="[
|
||||
selected
|
||||
? 'font-semibold text-zinc-100'
|
||||
: 'font-normal',
|
||||
'block truncate',
|
||||
]"
|
||||
>{{ label }}</span
|
||||
>
|
||||
|
||||
<span
|
||||
v-if="selected"
|
||||
:class="[
|
||||
active ? 'text-white' : 'text-blue-600',
|
||||
'absolute inset-y-0 right-0 flex items-center pr-4',
|
||||
]"
|
||||
>
|
||||
<CheckIcon class="h-5 w-5" aria-hidden="true" />
|
||||
</span>
|
||||
</li>
|
||||
</ListboxOption>
|
||||
</ListboxOptions>
|
||||
</transition>
|
||||
</div>
|
||||
</Listbox>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<label
|
||||
for="name"
|
||||
class="block text-sm font-medium leading-6 text-zinc-100"
|
||||
>{{ $t("account.token.acls") }}</label
|
||||
>
|
||||
<p class="text-zinc-400 block text-xs font-medium leading-6">
|
||||
{{ $t("account.token.aclsDesc") }}
|
||||
</p>
|
||||
<fieldset class="divide-y divide-zinc-700">
|
||||
<div
|
||||
v-for="[sectionName, sectionAcls] in Object.entries(
|
||||
aclsBySection,
|
||||
)"
|
||||
:key="sectionName"
|
||||
class="grid lg:grid-cols-3 gap-1 py-3"
|
||||
>
|
||||
<div
|
||||
v-for="[acl, description] in Object.entries(sectionAcls)"
|
||||
:key="acl"
|
||||
class="flex gap-3"
|
||||
>
|
||||
<div class="flex h-6 shrink-0 items-center">
|
||||
<div class="group grid size-4 grid-cols-1">
|
||||
<input
|
||||
id="acl"
|
||||
v-model="currentACLs[acl]"
|
||||
aria-describedby="acl-description"
|
||||
name="acl"
|
||||
type="checkbox"
|
||||
class="col-start-1 row-start-1 appearance-none rounded-sm border checked:border-blue-600 checked:bg-blue-600 indeterminate:border-blue-600 indeterminate:bg-blue-600 focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-blue-600 disabled:border-gray-300 disabled:bg-gray-100 disabled:checked:bg-gray-100 border-white/10 bg-white/5 dark:checked:border-blue-500 dark:checked:bg-blue-500 dark:indeterminate:border-blue-500 dark:indeterminate:bg-blue-500 dark:focus-visible:outline-blue-500 dark:disabled:border-white/5 dark:disabled:bg-white/10 dark:disabled:checked:bg-white/10 forced-colors:appearance-auto"
|
||||
/>
|
||||
<svg
|
||||
class="pointer-events-none col-start-1 row-start-1 size-3.5 self-center justify-self-center stroke-white group-has-disabled:stroke-white/25"
|
||||
viewBox="0 0 14 14"
|
||||
fill="none"
|
||||
>
|
||||
<path
|
||||
class="opacity-0 group-has-checked:opacity-100"
|
||||
d="M3 8L6 11L11 3.5"
|
||||
stroke-width="2"
|
||||
stroke-linecap="round"
|
||||
stroke-linejoin="round"
|
||||
/>
|
||||
<path
|
||||
class="opacity-0 group-has-indeterminate:opacity-100"
|
||||
d="M3 7H11"
|
||||
stroke-width="2"
|
||||
stroke-linecap="round"
|
||||
stroke-linejoin="round"
|
||||
/>
|
||||
</svg>
|
||||
</div>
|
||||
</div>
|
||||
<div class="text-sm/6">
|
||||
<label
|
||||
for="acl"
|
||||
class="font-display font-medium text-white"
|
||||
>{{ acl }}</label
|
||||
>
|
||||
{{ " " }}
|
||||
<span id="acl-description" class="text-xs text-zinc-400"
|
||||
><span class="sr-only">{{ acl }} </span
|
||||
>{{ description }}</span
|
||||
>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</fieldset>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
<template #buttons>
|
||||
<LoadingButton :loading="props.loading" @click="() => createToken()">
|
||||
{{ $t("common.create") }}
|
||||
</LoadingButton>
|
||||
<button
|
||||
class="inline-flex items-center rounded-md bg-zinc-800 px-3 py-2 text-sm font-semibold font-display text-white hover:bg-zinc-700"
|
||||
@click="() => cancel()"
|
||||
>
|
||||
{{ $t("cancel") }}
|
||||
</button>
|
||||
</template>
|
||||
</ModalTemplate>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import {
|
||||
Listbox,
|
||||
ListboxButton,
|
||||
ListboxLabel,
|
||||
ListboxOption,
|
||||
ListboxOptions,
|
||||
} from "@headlessui/vue";
|
||||
import { CheckIcon, ChevronUpDownIcon } from "@heroicons/vue/24/outline";
|
||||
import type { DurationLike } from "luxon";
|
||||
|
||||
// Reuse for both admin and user tokens
|
||||
|
||||
const model = defineModel<boolean>({ required: true });
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const props = defineProps<{
|
||||
acls: { [key: string]: string };
|
||||
loading?: boolean;
|
||||
suggestedAcls?: string[];
|
||||
suggestedName?: string;
|
||||
}>();
|
||||
|
||||
// Label to parameters to moment.js .add()
|
||||
const expiry: Record<string, DurationLike | undefined> = {
|
||||
[t("account.token.expiryMonth")]: {
|
||||
month: 1,
|
||||
},
|
||||
[t("account.token.expiry3Month")]: {
|
||||
month: 3,
|
||||
},
|
||||
[t("account.token.expiry6Month")]: {
|
||||
month: 6,
|
||||
},
|
||||
[t("account.token.expiryYear")]: {
|
||||
year: 1,
|
||||
},
|
||||
[t("account.token.expiry5Year")]: {
|
||||
year: 5,
|
||||
},
|
||||
[t("account.token.noExpiry")]: undefined,
|
||||
};
|
||||
const expiryKey = ref<keyof typeof expiry>(Object.keys(expiry)[0]); // Cast to any because we just know it's okay
|
||||
const name = ref(props.suggestedName ?? "");
|
||||
const currentACLs = ref<{ [key: string]: boolean }>(
|
||||
Object.fromEntries((props.suggestedAcls ?? []).map((v) => [v, true])),
|
||||
);
|
||||
|
||||
const aclsBySection = computed(() => {
|
||||
const sections: { [key: string]: { [key: string]: string } } = {};
|
||||
for (const [acl, description] of Object.entries(props.acls)) {
|
||||
const section = acl.split(":")[0];
|
||||
sections[section] ??= {};
|
||||
sections[section][acl] = description;
|
||||
}
|
||||
return sections;
|
||||
});
|
||||
|
||||
const emit = defineEmits<{
|
||||
create: [name: string, acls: string[], expiry: DurationLike | undefined];
|
||||
}>();
|
||||
|
||||
function createToken() {
|
||||
emit(
|
||||
"create",
|
||||
name.value,
|
||||
Object.entries(currentACLs.value)
|
||||
.filter(([_acl, enabled]) => enabled)
|
||||
.map(([acl, _enabled]) => acl),
|
||||
expiry[expiryKey.value],
|
||||
);
|
||||
}
|
||||
|
||||
function cancel() {
|
||||
model.value = false;
|
||||
}
|
||||
|
||||
watch(model, (c) => {
|
||||
if (!c) {
|
||||
name.value = "";
|
||||
currentACLs.value = {};
|
||||
}
|
||||
});
|
||||
</script>
|
||||
@ -176,9 +176,12 @@
|
||||
active ? 'bg-zinc-900 outline-hidden' : '',
|
||||
'w-full text-left block px-4 py-2 text-sm',
|
||||
]"
|
||||
@click="() => (currentSort = option.param)"
|
||||
@click.prevent="handleSortClick(option, $event)"
|
||||
>
|
||||
{{ option.name }}
|
||||
<span v-if="currentSort === option.param">
|
||||
{{ sortOrder === 'asc' ? '↑' : '↓' }}
|
||||
</span>
|
||||
</button>
|
||||
</MenuItem>
|
||||
</div>
|
||||
@ -389,8 +392,13 @@ const sorts: Array<StoreSortOption> = [
|
||||
name: "Recently Added",
|
||||
param: "recent",
|
||||
},
|
||||
{
|
||||
name: "Name",
|
||||
param: "name",
|
||||
},
|
||||
];
|
||||
const currentSort = ref(sorts[0].param);
|
||||
const sortOrder = ref<"asc" | "desc">("desc");
|
||||
|
||||
const options: Array<StoreFilterOption> = [
|
||||
...(tags.length > 0
|
||||
@ -466,7 +474,7 @@ async function updateGames(query: string, resetGames: boolean) {
|
||||
results: Array<SerializeObject<GameModel>>;
|
||||
count: number;
|
||||
}>(
|
||||
`/api/v1/store?take=50&skip=${resetGames ? 0 : games.value?.length || 0}&sort=${currentSort.value}${query ? "&" + query : ""}`,
|
||||
`/api/v1/store?take=50&skip=${resetGames ? 0 : games.value?.length || 0}&sort=${currentSort.value}&order=${sortOrder.value}${query ? "&" + query : ""}`,
|
||||
);
|
||||
if (resetGames) {
|
||||
games.value = newValues.results;
|
||||
@ -483,6 +491,20 @@ watch(filterQuery, (newUrl) => {
|
||||
watch(currentSort, (_) => {
|
||||
updateGames(filterQuery.value, true);
|
||||
});
|
||||
watch(sortOrder, (_) => {
|
||||
updateGames(filterQuery.value, true);
|
||||
});
|
||||
|
||||
await updateGames(filterQuery.value, true);
|
||||
</script>
|
||||
|
||||
function handleSortClick(option: StoreSortOption, event: MouseEvent) {
|
||||
event.stopPropagation();
|
||||
if (currentSort.value === option.param) {
|
||||
sortOrder.value = sortOrder.value === 'asc' ? 'desc' : 'asc';
|
||||
} else {
|
||||
currentSort.value = option.param;
|
||||
sortOrder.value = option.param === 'name' ? 'asc' : 'desc';
|
||||
}
|
||||
}
|
||||
|
||||
</script>
|
||||
55
components/TaskWidget.vue
Normal file
55
components/TaskWidget.vue
Normal file
@ -0,0 +1,55 @@
|
||||
<template>
|
||||
<div
|
||||
v-if="task"
|
||||
class="flex w-full items-center justify-between space-x-6 p-6"
|
||||
>
|
||||
<div class="flex-1 truncate">
|
||||
<div class="flex items-center space-x-1">
|
||||
<div>
|
||||
<CheckCircleIcon v-if="task.success" class="size-5 text-green-600" />
|
||||
<XMarkIcon v-else-if="task.error" class="size-5 text-red-600" />
|
||||
<div
|
||||
v-else
|
||||
class="size-2 bg-blue-600 rounded-full animate-pulse m-1"
|
||||
/>
|
||||
</div>
|
||||
<h3 class="truncate text-sm font-medium text-zinc-100">
|
||||
{{ task.name }}
|
||||
</h3>
|
||||
</div>
|
||||
<div
|
||||
v-if="active"
|
||||
class="mt-2 w-full rounded-full overflow-hidden bg-zinc-900"
|
||||
>
|
||||
<div
|
||||
:style="{ width: `${task.progress}%` }"
|
||||
class="bg-blue-600 h-[3px] transition-all"
|
||||
/>
|
||||
</div>
|
||||
<div class="mt-2 bg-zinc-950 px-2 pb-1 rounded-sm">
|
||||
<LogLine :short="true" :log="parseTaskLog(task.log.at(-1))" />
|
||||
</div>
|
||||
<NuxtLink
|
||||
type="button"
|
||||
:href="`/admin/task/${task.id}`"
|
||||
class="mt-3 ml-1 rounded-md text-xs font-medium text-zinc-100 hover:text-zinc-300 focus:outline-none focus:ring-2 focus:ring-zinc-100 focus:ring-offset-2"
|
||||
>
|
||||
<i18n-t keypath="tasks.admin.viewTask" tag="span" scope="global">
|
||||
<template #arrow>
|
||||
<span aria-hidden="true">{{ $t("chars.arrow") }}</span>
|
||||
</template>
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
</div>
|
||||
</div>
|
||||
<div v-else>
|
||||
<!-- renders server side when we don't want to access the current tasks -->
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { CheckCircleIcon, XMarkIcon } from "@heroicons/vue/24/solid";
|
||||
import type { TaskMessage } from "~/server/internal/tasks";
|
||||
|
||||
defineProps<{ task: TaskMessage | undefined; active?: boolean }>();
|
||||
</script>
|
||||
@ -91,10 +91,11 @@
|
||||
</div>
|
||||
|
||||
<div class="flex items-center justify-center xl:col-span-3 mt-8">
|
||||
<p
|
||||
<NuxtLink
|
||||
:to="`https://github.com/Drop-OSS/drop/releases/tag/${versionInfo.version}`"
|
||||
class="text-xs text-zinc-700 hover:text-zinc-400 transition-colors duration-200 cursor-default select-none"
|
||||
>
|
||||
<i18n-t keypath="footer.version" tag="p" scope="global">
|
||||
<i18n-t keypath="footer.version" tag="span" scope="global">
|
||||
<template #version>
|
||||
<span>{{ versionInfo.version }}</span>
|
||||
</template>
|
||||
@ -102,7 +103,7 @@
|
||||
<span>{{ versionInfo.gitRef }}</span>
|
||||
</template>
|
||||
</i18n-t>
|
||||
</p>
|
||||
</NuxtLink>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@ -46,10 +46,28 @@ export const $dropFetch: DropFetch = async (rawRequest, opts) => {
|
||||
});
|
||||
const request = requestParts.join("/");
|
||||
|
||||
// If not in setup
|
||||
if (!getCurrentInstance()?.proxy) {
|
||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||
// @ts-ignore Excessive stack depth comparing types
|
||||
return await $fetch(request, opts);
|
||||
try {
|
||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||
// @ts-ignore Excessive stack depth comparing types
|
||||
return await $fetch(request, opts);
|
||||
} catch (e) {
|
||||
if (import.meta.client && opts?.failTitle) {
|
||||
console.warn(e);
|
||||
createModal(
|
||||
ModalType.Notification,
|
||||
{
|
||||
title: opts.failTitle,
|
||||
description:
|
||||
(e as FetchError)?.statusMessage ?? (e as string).toString(),
|
||||
//buttonText: $t("common.close"),
|
||||
},
|
||||
(_, c) => c(),
|
||||
);
|
||||
}
|
||||
throw e;
|
||||
}
|
||||
}
|
||||
|
||||
const id = request.toString();
|
||||
@ -64,26 +82,10 @@ export const $dropFetch: DropFetch = async (rawRequest, opts) => {
|
||||
}
|
||||
|
||||
const headers = useRequestHeaders(["cookie", "authorization"]);
|
||||
try {
|
||||
const data = await $fetch(request, {
|
||||
...opts,
|
||||
headers: { ...headers, ...opts?.headers },
|
||||
});
|
||||
if (import.meta.server) state.value = data;
|
||||
return data;
|
||||
} catch (e) {
|
||||
if (import.meta.client && opts?.failTitle) {
|
||||
createModal(
|
||||
ModalType.Notification,
|
||||
{
|
||||
title: opts.failTitle,
|
||||
description:
|
||||
(e as FetchError)?.statusMessage ?? (e as string).toString(),
|
||||
buttonText: $t("common.close"),
|
||||
},
|
||||
(_, c) => c(),
|
||||
);
|
||||
}
|
||||
throw e;
|
||||
}
|
||||
const data = await $fetch(request, {
|
||||
...opts,
|
||||
headers: { ...headers, ...opts?.headers },
|
||||
});
|
||||
if (import.meta.server) state.value = data;
|
||||
return data;
|
||||
};
|
||||
|
||||
Submodule drop-base updated: 04125e89be...06bea06363
@ -1 +1,673 @@
|
||||
{}
|
||||
{
|
||||
"account": {
|
||||
"devices": {
|
||||
"capabilities": "Funktionen",
|
||||
"lastConnected": "Zuletzt verbunden",
|
||||
"noDevices": "Keine Geräte sind mit deinem Konto verbunden.",
|
||||
"platform": "Plattform",
|
||||
"revoke": "Wiederrufen",
|
||||
"subheader": "Geräte verwalten, die auf Ihr Drop Konto zugreifen dürfen.",
|
||||
"title": "Geräte"
|
||||
},
|
||||
"notifications": {
|
||||
"all": "Alles anzeigen {arrow}",
|
||||
"desc": "Anzeigen und Verwalten deiner Benachrichtigung.",
|
||||
"markAllAsRead": "Markiere alle als gelesen",
|
||||
"markAsRead": "Als gelesen Markieren",
|
||||
"none": "Keine Benachrichtigungen",
|
||||
"notifications": "Benachrichtigungen",
|
||||
"title": "Benachrichtigungen",
|
||||
"unread": "Ungelesene Benachrichtigungen"
|
||||
},
|
||||
"settings": "Einstellungen",
|
||||
"title": "Kontoeinstellungen",
|
||||
"token": {
|
||||
"acls": "Berechtigungen (ACLs/Scopes)",
|
||||
"aclsDesc": "Definiert, wozu dieses Schlüssel berechtigt ist. Du solltest vermeiden, alle ACLs auszuwählen, wenn dies nicht notwendig ist.",
|
||||
"expiry": "Ablaufdatum",
|
||||
"expiry3Month": "3 Monate",
|
||||
"expiry5Year": "5 Jahre",
|
||||
"expiry6Month": "6 Monate",
|
||||
"expiryMonth": "Ein Monat",
|
||||
"expiryYear": "Ein Jahr",
|
||||
"name": "API-Schlüssel Name",
|
||||
"nameDesc": "Der Name des Schlüssels, als Referenz.",
|
||||
"namePlaceholder": "Mein neuer Schlüssel",
|
||||
"noExpiry": "Unbegrenzt gültig",
|
||||
"noTokens": "Keine Schlüssel mit deinem Konto verbunden.",
|
||||
"revoke": "Wiederrufen",
|
||||
"subheader": "Verwalte deine API-Schlüssel und deren Zugriffsrechte.",
|
||||
"success": "Schlüssel erfolgreich erstellt.",
|
||||
"successNote": "Bitte jetzt kopieren, da es nicht noch einmal angezeigt wird.",
|
||||
"title": "API-Schlüssel"
|
||||
}
|
||||
},
|
||||
"actions": "Aktionen",
|
||||
"add": "Hinzufügen",
|
||||
"adminTitle": "Admin Dashboard - Drop",
|
||||
"adminTitleTemplate": "{0} - Admin - Drop",
|
||||
"auth": {
|
||||
"callback": {
|
||||
"authClient": "Client autorisieren?",
|
||||
"authorize": "Autorisieren",
|
||||
"authorizedClient": "Drop hat den Client erfolgreich autorisiert. Du kannst dieses Fenster nun schließen.",
|
||||
"issues": "Probleme?",
|
||||
"learn": "Mehr erfahren {arrow}",
|
||||
"paste": "Füge diesen Code in den Client ein, um fortzufahren:",
|
||||
"permWarning": "Das akzeptieren dieser Anfrage erlaubt \"{name}\" auf \"{plattform}\" folgende Berechtigungen:",
|
||||
"requestedAccess": "\"{name}\" hat Zugriff auf dein Drop Konto angefordert.",
|
||||
"success": "Erfolgreich!"
|
||||
},
|
||||
"code": {
|
||||
"description": "Verwende einen Code, um dein Drop Client zu verbinden, wenn dein Gerät kein Webbrowser öffnen kann.",
|
||||
"title": "Verbinde deinen Drop Client"
|
||||
},
|
||||
"confirmPassword": "Bestätige @:auth.password",
|
||||
"displayName": "Anzeigename",
|
||||
"email": "E-Mail",
|
||||
"password": "Passwort",
|
||||
"register": {
|
||||
"confirmPasswordFormat": "Muss mit oben genanntem übereinstimmen",
|
||||
"emailFormat": "Muss im Format nutzer{'@'}beispiel.de sein",
|
||||
"passwordFormat": "Muss mindestens 14 Zeichen enthalten",
|
||||
"subheader": "Gebe unten deine Daten ein, um dein Konto zu erstellen.",
|
||||
"title": "Erstelle dein Drop Konto",
|
||||
"usernameFormat": "Muss mindestens 5 Zeichen enthalten und aus Kleinbuchstaben bestehen"
|
||||
},
|
||||
"signin": {
|
||||
"externalProvider": "Bei externem Anbieter anmelden {arrow}",
|
||||
"forgot": "Passwort vergessen?",
|
||||
"noAccount": "Noch kein Konto? Bitte den Admin, eines für dich zu erstellen.",
|
||||
"or": "ODER",
|
||||
"pageTitle": "Bei Drop anmelden",
|
||||
"rememberMe": "Erinnere mich",
|
||||
"signin": "Anmelden",
|
||||
"title": "Melde dich bei deinem Konto an"
|
||||
},
|
||||
"signout": "Ausloggen",
|
||||
"username": "Nutzername"
|
||||
},
|
||||
"cancel": "Abbrechen",
|
||||
"chars": {
|
||||
"arrow": "→",
|
||||
"arrowBack": "←",
|
||||
"quoted": "\"\"",
|
||||
"srComma": ", {0}"
|
||||
},
|
||||
"common": {
|
||||
"add": "Hinzufügen",
|
||||
"cannotUndo": "Diese Aktion kann nicht rückgängig gemacht werden.",
|
||||
"close": "Schließen",
|
||||
"create": "Erstellen",
|
||||
"date": "Datum",
|
||||
"deleteConfirm": "Möchtest du \"{0}\" wirklich löschen?",
|
||||
"divider": "{'|'}",
|
||||
"edit": "Bearbeiten",
|
||||
"friends": "Freunde",
|
||||
"groups": "Gruppen",
|
||||
"insert": "Einfügen",
|
||||
"name": "Name",
|
||||
"noResults": "Keine Ergebnisse",
|
||||
"noSelected": "Keine Elemente ausgewählt.",
|
||||
"remove": "Entfernen",
|
||||
"save": "Speichern",
|
||||
"saved": "Gespeichert",
|
||||
"servers": "Server",
|
||||
"srLoading": "Lade…",
|
||||
"tags": "Tags",
|
||||
"today": "Heute"
|
||||
},
|
||||
"delete": "Löschen",
|
||||
"drop": {
|
||||
"desc": "Eine Open-Source-Plattform für die Verteilung von Spielen, die auf Geschwindigkeit, Flexibilität und Ästhetik ausgelegt ist.",
|
||||
"drop": "Drop"
|
||||
},
|
||||
"editor": {
|
||||
"bold": "Fett",
|
||||
"boldPlaceholder": "fettgedruckter Text",
|
||||
"code": "Code",
|
||||
"codePlaceholder": "code",
|
||||
"heading": "Überschrift",
|
||||
"headingPlaceholder": "überschrift",
|
||||
"italic": "Kursiv",
|
||||
"italicPlaceholder": "kursiver Text",
|
||||
"link": "Link",
|
||||
"linkPlaceholder": "link Text",
|
||||
"listItem": "Listenelement",
|
||||
"listItemPlaceholder": "listenelement"
|
||||
},
|
||||
"errors": {
|
||||
"admin": {
|
||||
"user": {
|
||||
"delete": {
|
||||
"desc": "Drop konnte diesen Benutzer nicht löschen: {0}",
|
||||
"title": "Benutzer konnte nicht gelöscht werden"
|
||||
}
|
||||
}
|
||||
},
|
||||
"auth": {
|
||||
"disabled": "Ungültiges oder deaktiviertes Konto. Bitte kontaktiere einen Server Admin.",
|
||||
"invalidInvite": "Ungültige oder abgelaufene Einladung",
|
||||
"invalidPassState": "Ungültiger Passwortzustand. Bitte kontaktiere einen Server Admin.",
|
||||
"invalidUserOrPass": "Ungültiger Nutzername oder Passwort.",
|
||||
"inviteIdRequired": "id erforderlich beim Abrufen der Einladung",
|
||||
"method": {
|
||||
"signinDisabled": "Anmeldemethode nicht aktiviert"
|
||||
},
|
||||
"usernameTaken": "Nutzername bereits vergeben."
|
||||
},
|
||||
"backHome": "{arrow} Zurück zur Startseite",
|
||||
"externalUrl": {
|
||||
"subtitle": "Diese Nachricht ist nur sichtbar für Admins.",
|
||||
"title": "Zugriff über eine andere EXTERNAL_URL. Bitte die Dokumentation prüfen."
|
||||
},
|
||||
"game": {
|
||||
"banner": {
|
||||
"description": "Das Aktualisieren des Banners ist fehlgeschlagen: {0}",
|
||||
"title": "Das Aktualisieren des Banners ist fehlgeschlagen"
|
||||
},
|
||||
"carousel": {
|
||||
"description": "Das Aktualisieren des Bildkarussells ist fehlgeschlagen: {0}",
|
||||
"title": "Das Aktualisieren des Bildkarussells ist fehlgeschlagen"
|
||||
},
|
||||
"cover": {
|
||||
"description": "Das Aktualisieren des Titelbildes ist fehlgeschlagen: {0}",
|
||||
"title": "Das Aktualisieren des Titelbildes ist fehlgeschlagen"
|
||||
},
|
||||
"deleteImage": {
|
||||
"description": "Das Löschen des Bildes ist fehlgeschlagen: {0}",
|
||||
"title": "Das Löschen des Bildes ist fehlgeschlagen"
|
||||
},
|
||||
"description": {
|
||||
"description": "Das Aktualisieren der Spielbeschreibung ist fehlgeschlagen: {0}",
|
||||
"title": "Das Aktualisieren der Spielbeschreibung ist fehlgeschlagen"
|
||||
},
|
||||
"metadata": {
|
||||
"description": "Das Aktualisieren der Spielmetadaten ist fehlgeschlagen: {0}",
|
||||
"title": "Das Aktualisieren der Spielmetadaten ist fehlgeschlagen"
|
||||
}
|
||||
},
|
||||
"invalidBody": "Ungültiger Anfragenkörper: {0}",
|
||||
"inviteRequired": "Registrierung nur mit Einladung möglich.",
|
||||
"library": {
|
||||
"add": {
|
||||
"desc": "Drop konnte dieses Spiel nicht zu deiner Bibliothek hinzufügen: {0}",
|
||||
"title": "Das Spiel konnte nicht zur Bibliothek hinzugefügt werden"
|
||||
},
|
||||
"collection": {
|
||||
"create": {
|
||||
"desc": "Das Erstellen der Sammlung ist fehlgeschlagen: {0}",
|
||||
"title": "Das Erstellen der Sammlung ist fehlgeschlagen"
|
||||
}
|
||||
},
|
||||
"source": {
|
||||
"delete": {
|
||||
"desc": "Das Löschen der Quelle ist fehlgeschlagen: {0}",
|
||||
"title": "Das Löschen der Quellbibliothek ist fehlgeschlagen"
|
||||
}
|
||||
}
|
||||
},
|
||||
"news": {
|
||||
"article": {
|
||||
"delete": {
|
||||
"desc": "Das Löschen des Artikels ist fehlgeschlagen: {0}",
|
||||
"title": "Das Löschen des Artikels ist fehlgeschlagen"
|
||||
}
|
||||
}
|
||||
},
|
||||
"occurred": "Bei der Bearbeitung deiner Anfrage ist ein Fehler aufgetreten. Wenn du glaubst, dass es sich um einen Bug handelt, melde diesen bitte. Versuche dich anzumelden, um zu sehen, ob dadurch das Problem behoben wird.",
|
||||
"ohNo": "Oh nein!",
|
||||
"pageTitle": "{0} | Drop",
|
||||
"revokeClient": "Client konnte nicht widerrufen werden",
|
||||
"revokeClientFull": "Client konnte nicht widerrufen werden {0}",
|
||||
"signIn": "Anmelden {arrow}",
|
||||
"support": "Support Discord",
|
||||
"unknown": "Ein unbekannter Fehler ist aufgetreten",
|
||||
"upload": {
|
||||
"description": "Drop konnte die Datei nicht hochladen: {0}",
|
||||
"title": "Das hochladen der Datei ist Fehlgeschlagen"
|
||||
},
|
||||
"version": {
|
||||
"delete": {
|
||||
"desc": "Beim Löschen der Version ist ein Fehler aufgetreten: {error}",
|
||||
"title": "Beim Löschen der Version ist ein Fehler aufgetreten"
|
||||
},
|
||||
"order": {
|
||||
"desc": "Beim Aktualisieren der Version ist ein Fehler aufgetreten: {error}",
|
||||
"title": "Beim Aktualisieren der Versionsreihenfolge ist ein Fehler aufgetreten"
|
||||
}
|
||||
}
|
||||
},
|
||||
"footer": {
|
||||
"about": "Über",
|
||||
"aboutDrop": "Über Drop",
|
||||
"comparison": "Vergleich",
|
||||
"docs": {
|
||||
"client": "Client Dokumentation",
|
||||
"server": "Server Dokumentation"
|
||||
},
|
||||
"documentation": "Dokumentation",
|
||||
"findGame": "Finde ein Spiel",
|
||||
"footer": "Fußzeile",
|
||||
"games": "Spiels",
|
||||
"social": {
|
||||
"discord": "Discord",
|
||||
"github": "GitHub"
|
||||
},
|
||||
"topSellers": "Bestseller",
|
||||
"version": "Drop {version} {gitRef}"
|
||||
},
|
||||
"header": {
|
||||
"admin": {
|
||||
"admin": "Admin",
|
||||
"metadata": "Metadaten",
|
||||
"settings": {
|
||||
"store": "Store",
|
||||
"title": "Einstellungen",
|
||||
"tokens": "API-Schlüssel"
|
||||
},
|
||||
"tasks": "Aufgaben",
|
||||
"users": "Benutzer"
|
||||
},
|
||||
"back": "Zurück",
|
||||
"openSidebar": "Öffne Seitenleiste"
|
||||
},
|
||||
"helpUsTranslate": "Hilf uns Drop zu übersetzen {arrow}",
|
||||
"highest": "Höchste",
|
||||
"home": "Startseite",
|
||||
"library": {
|
||||
"addGames": "Alle Spiele",
|
||||
"addToLib": "Zur Bibliothek hinzufügen",
|
||||
"admin": {
|
||||
"detectedGame": "Drop hat erkannt, dass du ein neues Spiel importieren kannst.",
|
||||
"detectedVersion": "Drop hat erkannt, dass du eine neue Version dieses Spiels importieren kannst.",
|
||||
"game": {
|
||||
"addCarouselNoImages": "Keine Bilder zum hinzufügen.",
|
||||
"addDescriptionNoImages": "Keine Bilder zum hinzufügen.",
|
||||
"addImageCarousel": "Aus der Bilderbibliothek hinzufügen",
|
||||
"currentBanner": "Banner",
|
||||
"currentCover": "Cover",
|
||||
"deleteImage": "Bild löschen",
|
||||
"editGameDescription": "Spielbeschreibung",
|
||||
"editGameName": "Spielname",
|
||||
"imageCarousel": "Bilderkarussell",
|
||||
"imageCarouselDescription": "Anpassen, welche Bilder und in welcher Reihenfolge sie auf der Shop-Seite angezeigt werden.",
|
||||
"imageCarouselEmpty": "Es wurden noch keine Bilder zum Karussell hinzugefügt.",
|
||||
"imageLibrary": "Bilderbibliothek",
|
||||
"imageLibraryDescription": "Bitte beachten: Alle hochgeladenen Bilder sind für alle Nutzer über die Browser-Entwicklertools zugänglich.",
|
||||
"removeImageCarousel": "Bild entfernen",
|
||||
"setBanner": "Als Banner festlegen",
|
||||
"setCover": "Als Cover festlegen"
|
||||
},
|
||||
"gameLibrary": "Spielebibliothek",
|
||||
"import": {
|
||||
"bulkImportDescription": "Auf dieser Seite wirst du nicht zur Importaufgabe weitergeleitet, sodass du mehrere Spiele nacheinander importieren kannst.",
|
||||
"bulkImportTitle": "Massenimport Modus",
|
||||
"import": "Import",
|
||||
"link": "Import {arrow}",
|
||||
"loading": "Spieldaten werden geladen…",
|
||||
"search": "Suche",
|
||||
"searchPlaceholder": "Fallout 4",
|
||||
"selectDir": "Bitte wähle ein Verzeichnis aus…",
|
||||
"selectGame": "Spiel zum Import auswählen",
|
||||
"selectGamePlaceholder": "Bitte wähle ein Spiel aus…",
|
||||
"selectGameSearch": "Spiel auswählen",
|
||||
"selectPlatform": "Bitte wähle eine Plattform aus…",
|
||||
"version": {
|
||||
"advancedOptions": "Erweiterte Optionen",
|
||||
"import": "Version Importieren",
|
||||
"installDir": "(Installationsverzeichnis)/",
|
||||
"launchCmd": "Programm/Befehl starten",
|
||||
"launchDesc": "Ausführbare Datei zum starten des Spiels",
|
||||
"launchPlaceholder": "spiel.exe",
|
||||
"loadingVersion": "Lade Versionsmetadaten…",
|
||||
"noAdv": "Keine erweiterten Optionen für diese Konfiguration.",
|
||||
"noVersions": "Keine Version zum importieren",
|
||||
"platform": "Plattformversion",
|
||||
"setupCmd": "Installationsprogramm oder Befehl ausführen",
|
||||
"setupDesc": "Wird einmal ausgeführt, wenn das Spiel installiert wird",
|
||||
"setupMode": "Einrichtungsmodus",
|
||||
"setupModeDesc": "Wenn aktiviert, hat diese Version keinen Startbefehl und führt einfach die ausführbare Datei auf dem Computer des Nutzers aus. Nützlich für Spiele, die nur einen Installer bereitstellen und keine portablen Dateien.",
|
||||
"setupPlaceholder": "setup.exe",
|
||||
"umuLauncherId": "UMU Launcher ID",
|
||||
"umuOverride": "Überschreibe UMU Launcher Spiel ID",
|
||||
"umuOverrideDesc": "Standardmäßig verwendet Drop beim Start über den UMU Launcher eine Nicht-ID. Um die richtigen Patches für manche Spiele zu erhalten, musst du dieses Feld eventuell manuell setzen.",
|
||||
"updateMode": "Aktualisierungsmodus",
|
||||
"updateModeDesc": "Wenn aktiviert, werden diese Dateien über die vorherige Version installiert (überschrieben). Werden mehrere ‚Update-Modi‘ hintereinander verwendet, werden sie in der angegebenen Reihenfolge angewendet.",
|
||||
"version": "Wähle die Version für den Import aus"
|
||||
},
|
||||
"withoutMetadata": "Ohne Metadaten importieren"
|
||||
},
|
||||
"libraryHint": "Keine Bibliotheken konfiguriert.",
|
||||
"libraryHintDocsLink": "Was bedeutet das? {arrow}",
|
||||
"metadata": {
|
||||
"companies": {
|
||||
"action": "Verwalte {arrow}",
|
||||
"addGame": {
|
||||
"description": "Wähle ein Spiel aus, das dem Unternehmen hinzugefügt werden soll, und lege fest, ob es als Entwickler, Publisher oder beides geführt werden soll.",
|
||||
"developer": "Entwickler?",
|
||||
"noGames": "Keine Spiele zum hinzufügen",
|
||||
"publisher": "Publisher?",
|
||||
"title": "Verbinde das Spiel mit diesem Unternehmen"
|
||||
},
|
||||
"description": "Unternehmen organisieren Spiele danach, wer sie entwickelt oder veröffentlicht hat.",
|
||||
"editor": {
|
||||
"action": "Spiel hinzufügen {plus}",
|
||||
"descriptionPlaceholder": "{'<'}Beschreibung{'>'}",
|
||||
"developed": "Entwickelt",
|
||||
"libraryDescription": "Hinzufügen, bearbeiten oder entfernen, was diese Firma entwickelt und/oder veröffentlicht hat.",
|
||||
"libraryTitle": "Spielebibliothek",
|
||||
"noDescription": "(Keine Beschreibung)",
|
||||
"published": "Veröffentlicht",
|
||||
"uploadBanner": "Banner hochladen",
|
||||
"uploadIcon": "Icon hochladen",
|
||||
"websitePlaceholder": "{'<'}Webseite{'>'}"
|
||||
},
|
||||
"modals": {
|
||||
"createDescription": "Erstelle ein Unternehmen, um deine Spiele besser zu organisieren.",
|
||||
"createFieldDescription": "Unternehmensbeschreibung",
|
||||
"createFieldDescriptionPlaceholder": "Ein kleines Indie-Studio, das…",
|
||||
"createFieldName": "Unternehmensname",
|
||||
"createFieldNamePlaceholder": "Mein neues Unternehmen…",
|
||||
"createFieldWebsite": "Unternehmenswebseite",
|
||||
"createFieldWebsitePlaceholder": "https://beispiel.de/",
|
||||
"createTitle": "Unternehmen erstellen",
|
||||
"nameDescription": "Bearbeite den Namen des Unternehmens. Wird verwendet, um neue Spielimporte zuzuordnen.",
|
||||
"nameTitle": "Bearbeite Firmenname",
|
||||
"shortDeckDescription": "Bearbeite die Firmenbeschreibung. Beeinträchtigt nicht die Lange (markdown) Beschreibung.",
|
||||
"shortDeckTitle": "Bearbeite Firmenbeschreibung",
|
||||
"websiteDescription": "„Bearbeite die Webseite des Unternehmens. Hinweis: Dies wird ein Link sein und bietet keinen Redirect-Schutz.",
|
||||
"websiteTitle": "Unternehmenswebseite bearbeiten"
|
||||
},
|
||||
"noCompanies": "Keine Unternehmen",
|
||||
"noGames": "Keine Spiele",
|
||||
"search": "Suche Unternehmen…",
|
||||
"searchGames": "Unternehmensspiele durchsuchen…",
|
||||
"title": "Unternehmen"
|
||||
},
|
||||
"tags": {
|
||||
"action": "Verwalte {arrow}",
|
||||
"create": "Erstellen",
|
||||
"description": "Tags werden automatisch aus importierten Genres erstellt. Du kannst eigene Tags hinzufügen, um deine Spielbibliothek zu kategorisieren.",
|
||||
"modal": {
|
||||
"description": "Erstelle einen Tag, um deine Bibliothek zu organisieren.",
|
||||
"title": "Tag erstellen"
|
||||
},
|
||||
"title": "Tags"
|
||||
}
|
||||
},
|
||||
"metadataProvider": "Metadatenanbieter",
|
||||
"noGames": "Keine Spiele importiert",
|
||||
"offline": "Drop konnte auf dieses Spiel nicht zugreifen.",
|
||||
"offlineTitle": "Spiel offline",
|
||||
"openEditor": "Im Editor öffnen {arrow}",
|
||||
"openStore": "Im Store öffnen",
|
||||
"shortDesc": "Kurzbeschreibung",
|
||||
"sources": {
|
||||
"create": "Quelle erstellen",
|
||||
"createDesc": "Drop wird diese Quelle verwenden, um auf deine Spielbibliothek zuzugreifen und die Spiele verfügbar zu machen.",
|
||||
"desc": "Konfiguriere deine Bibliotheksquellen, wo Drop nach neuen Spielen und Versionen zum Import suchen wird.",
|
||||
"documentationLink": "Dokumentation {arrow}",
|
||||
"edit": "Quelle bearbeiten",
|
||||
"fsDesc": "Importiert Spiele von einem Pfad auf der Festplatte. Benötigt eine versionsbasierte Ordnerstruktur und unterstützt archivierte Spiele.",
|
||||
"fsFlatDesc": "Importiert Spiele von einem Pfad auf der Festplatte, jedoch ohne separate Unterordner für Versionen. Nützlich beim Migrieren einer bestehenden Bibliothek zu Drop.",
|
||||
"fsFlatTitle": "Kompatibilität",
|
||||
"fsPath": "Pfad",
|
||||
"fsPathDesc": "Absoluter Pfad zur Spielebibliothek.",
|
||||
"fsPathPlaceholder": "/mnt/spiele",
|
||||
"fsTitle": "Drop-Stil",
|
||||
"link": "Quellen {arrow}",
|
||||
"nameDesc": "Der Name deiner Quelle, als Referenz.",
|
||||
"namePlaceholder": "Meine neue Quelle",
|
||||
"sources": "Bibliotheksquellen",
|
||||
"typeDesc": "Der Typ deiner Quelle. Ändert die erforderlichen Optionen.",
|
||||
"working": "Funktioniert es?"
|
||||
},
|
||||
"subheader": "Wenn du Ordner zu deinen Bibliotheksquellen hinzufügst, erkennt Drop diese und fordert dich auf, sie zu importieren. Jedes Spiel muss importiert werden, bevor du eine Version importieren kannst.",
|
||||
"title": "Bibliotheken",
|
||||
"version": {
|
||||
"delta": "Upgrade Modus",
|
||||
"noVersions": "Du hast keine verfügbare Version dieses Spiels.",
|
||||
"noVersionsAdded": "keine Versionen hinzugefügt"
|
||||
},
|
||||
"versionPriority": "Versions Priorität"
|
||||
},
|
||||
"back": "Zurück zur Bibliothek",
|
||||
"collection": {
|
||||
"addToNew": "Zur neuen Sammlung hinzufügen",
|
||||
"collections": "Sammlungen",
|
||||
"create": "Sammlung erstellen",
|
||||
"createDesc": "Sammlungen können genutzt werden, um deine Spiele zu organisieren und sie einfacher zu finden. Besonders bei großen Bibliotheken.",
|
||||
"delete": "Sammlung löschen",
|
||||
"namePlaceholder": "Sammlungsname",
|
||||
"noCollections": "Keine Sammlungen",
|
||||
"notFound": "Sammlung nicht gefunden",
|
||||
"subheader": "Füge eine neue Sammlung hinzu, um deine Spiele zu organisieren",
|
||||
"title": "Sammlung"
|
||||
},
|
||||
"gameCount": "{0} Spiele | {0} Spiele | {0} Spiele",
|
||||
"inLib": "In der Bibliothek",
|
||||
"launcherOpen": "Im Launcher öffnen",
|
||||
"noGames": "Keine Spiele in der Bibliothek",
|
||||
"notFound": "Spiel nicht gefunden",
|
||||
"search": "Durchsuche Bibliothek…",
|
||||
"subheader": "Verwalte deine Spiele in Sammlungen für einen einfacheren Zugriff auf alle deine Spiele."
|
||||
},
|
||||
"lowest": "Niedrigste",
|
||||
"news": {
|
||||
"article": {
|
||||
"add": "Hinzufügen",
|
||||
"content": "Inhalt (Markdown)",
|
||||
"create": "Neuen Artikel erstellen",
|
||||
"editor": "Editor",
|
||||
"editorGuide": "Verwende die obigen Shortcuts oder schreibe direkt in Markdown. Unterstützt **fett**, *kursiv*, [Links](URL) und mehr.",
|
||||
"new": "Neuer Artikel",
|
||||
"preview": "Vorschau",
|
||||
"shortDesc": "Kurzbeschreibung",
|
||||
"submit": "Absenden",
|
||||
"tagPlaceholder": "Tag hinzufügen…",
|
||||
"titles": "Titel",
|
||||
"uploadCover": "Cover hochladen"
|
||||
},
|
||||
"back": "Zurück zu Neuigkeiten",
|
||||
"checkLater": "Schaue später für Updates vorbei.",
|
||||
"delete": "Artikel löschen",
|
||||
"filter": {
|
||||
"all": "Gesamt",
|
||||
"month": "Diesen Monat",
|
||||
"week": "Diese Woche",
|
||||
"year": "Dieses Jahr"
|
||||
},
|
||||
"none": "Keine Artikel",
|
||||
"notFound": "Artikel nicht gefunden",
|
||||
"search": "Suche Artikel",
|
||||
"searchPlaceholder": "Suche Artikel…",
|
||||
"subheader": "Bleibe auf dem Laufenden über die neuesten Updates und Ankündigungen.",
|
||||
"title": "Neueste Neuigkeiten"
|
||||
},
|
||||
"options": "Einstellungen",
|
||||
"security": "Sicherheit",
|
||||
"selectLanguage": "Sprache auswählen",
|
||||
"settings": {
|
||||
"admin": {
|
||||
"description": "Konfiguriere Drop Einstellungen",
|
||||
"store": {
|
||||
"dropGameAltPlaceholder": "Beispiel Spielsymbol",
|
||||
"dropGameDescriptionPlaceholder": "Dies ist ein exemplarisches Spiel. Es wird ersetzt wenn du ein Spiel importierst.",
|
||||
"dropGameNamePlaceholder": "Beispielspiel",
|
||||
"showGamePanelTextDecoration": "Zeige Titel und Beschreibung auf den Spielkacheln (Standard: an)",
|
||||
"title": "Store"
|
||||
},
|
||||
"title": "Einstellungen"
|
||||
}
|
||||
},
|
||||
"setup": {
|
||||
"auth": {
|
||||
"description": "Die Authentifizierung in Drop erfolgt über mehrere konfigurierte ‚Provider‘. Jeder Provider ermöglicht es Nutzern, sich über seine Methode anzumelden. Um zu starten, sollte mindestens ein Authentifizierungs-Provider aktiviert sein und ein Konto über diesen erstellt werden.",
|
||||
"docs": "Dokumentation {arrow}",
|
||||
"enabled": "Aktiviert?",
|
||||
"openid": {
|
||||
"description": "OpenID Connect (OIDC) ist eine oft unterstützte OAuth2 Erweiterung. Drop erfordert die Konfiguration von OIDC über Umgebungsvariablen.",
|
||||
"skip": "Ich habe ein OIDC Nutzer",
|
||||
"title": "OpenID Connect"
|
||||
},
|
||||
"simple": {
|
||||
"description": "Die einfache Authentifizierung verwendet Nutzername und Password zur Authentifizierung von Benutzern. Sie ist standartmäßig aktiviert, wenn kein anderer Authentifizierungsanbieter aktiviert ist.",
|
||||
"register": "Als Admin registrieren {arrow}",
|
||||
"title": "Einfache Authentifizierung"
|
||||
},
|
||||
"title": "Authentifizierung"
|
||||
},
|
||||
"finish": "Los geht's {arrow}",
|
||||
"noPage": "keine Seite",
|
||||
"stages": {
|
||||
"account": {
|
||||
"description": "Du benötigst mindestens ein Konto, um Drop zu benutzen.",
|
||||
"name": "Richte dein Administratorkonto ein."
|
||||
},
|
||||
"library": {
|
||||
"description": "Füge mindestens eine Bibliotheksquelle hinzu, um Drop zu nutzen.",
|
||||
"name": "Erstelle eine Bibliothek."
|
||||
}
|
||||
},
|
||||
"welcome": "Hallo.",
|
||||
"welcomeDescription": "Willkommen zum Drop Einrichtungsassistenten. Er führt dich durch die erstmalige Konfiguration von Drop und erklärt dir, wie es funktioniert."
|
||||
},
|
||||
"store": {
|
||||
"about": "Über",
|
||||
"commingSoon": "Demnächst verfügbar",
|
||||
"developers": "Entwickler | Entwickler | Entwickler",
|
||||
"exploreMore": "Mehr entdecken {arrow}",
|
||||
"featured": "Empfohlen",
|
||||
"images": "Spielbilder",
|
||||
"lookAt": "Schau es dir an",
|
||||
"noDevelopers": "Keine Entwickler",
|
||||
"noFeatured": "KEINE HERVORGEHOBENEN SPIELE",
|
||||
"noGame": "KEIN SPIEL",
|
||||
"noImages": "Keine Bilder",
|
||||
"noPublishers": "Kein Publisher.",
|
||||
"noTags": "Keine Tags",
|
||||
"openAdminDashboard": "Im Admin Dashboard öffnen",
|
||||
"openFeatured": "Spiele in der Admin-Bibliothek markieren {arrow}",
|
||||
"platform": "Plattform | Plattform | Plattform",
|
||||
"publishers": "Publisher | Publisher | Publisher",
|
||||
"rating": "Bewertung",
|
||||
"readLess": "Weniger anzeigen",
|
||||
"readMore": "Mehr anzeigen",
|
||||
"recentlyAdded": "Kürzlich hinzugefügt",
|
||||
"recentlyReleased": "Kürzlich veröffentlicht",
|
||||
"recentlyUpdated": "Kürzlich aktualisiert",
|
||||
"released": "Veröffentlicht",
|
||||
"reviews": "({0} Bewertungen)",
|
||||
"tags": "Tags",
|
||||
"title": "Store",
|
||||
"view": {
|
||||
"sort": "Sortieren",
|
||||
"srFilters": "Filter",
|
||||
"srGames": "Spiele",
|
||||
"srViewGrid": "Raster anzeigen"
|
||||
},
|
||||
"viewInStore": "Im Store ansehen",
|
||||
"website": "Webseite"
|
||||
},
|
||||
"tasks": {
|
||||
"admin": {
|
||||
"back": "{arrow} Zurück zu den Aufgaben",
|
||||
"completedTasksTitle": "Abgeschlossene Aufgaben",
|
||||
"dailyScheduledTitle": "Tägliche Aufgaben",
|
||||
"execute": "{arrow} Ausführen",
|
||||
"noTasksRunning": "Keine laufenden Aufgaben",
|
||||
"progress": "{0}%",
|
||||
"runningTasksTitle": "Laufende Aufgaben",
|
||||
"scheduled": {
|
||||
"checkUpdateDescription": "Drop auf Updates überprüfen.",
|
||||
"checkUpdateName": "Auf Updates prüfen.",
|
||||
"cleanupInvitationsDescription": "Bereinigt abgelaufene Einladungen aus der Datenbank, um Speicherplatz zu sparen.",
|
||||
"cleanupInvitationsName": "Einladungen bereinigen",
|
||||
"cleanupObjectsDescription": "Erkennt und löscht nicht referenzierte und ungenutzte Objekte, um Speicherplatz zu sparen.",
|
||||
"cleanupObjectsName": "Objekte bereinigen",
|
||||
"cleanupSessionsDescription": "Bereinigt abgelaufene Sitzungen, um Speicherplatz zu sparen und die Sicherheit zu gewährleisten.",
|
||||
"cleanupSessionsName": "Sitzungen bereinigen."
|
||||
},
|
||||
"viewTask": "Ansehen {arrow}",
|
||||
"weeklyScheduledTitle": "Wöchentliche Aufgaben"
|
||||
}
|
||||
},
|
||||
"title": "Drop",
|
||||
"titleTemplate": "{0} - Drop",
|
||||
"todo": "Todo",
|
||||
"type": "Typ",
|
||||
"upload": "Hochladen",
|
||||
"uploadFile": "Datei hochladen",
|
||||
"userHeader": {
|
||||
"closeSidebar": "Seitenleiste schließen",
|
||||
"links": {
|
||||
"community": "Community",
|
||||
"library": "Bibliothek",
|
||||
"news": "Neuigkeiten"
|
||||
},
|
||||
"profile": {
|
||||
"admin": "Admin Dashboard",
|
||||
"settings": "Kontoeinstellungen"
|
||||
}
|
||||
},
|
||||
"users": {
|
||||
"admin": {
|
||||
"adminHeader": "Administrator?",
|
||||
"adminUserLabel": "Admin",
|
||||
"authLink": "Authentifizierung {arrow}",
|
||||
"authentication": {
|
||||
"configure": "Konfigurieren",
|
||||
"description": "Drop unterstützt eine Vielzahl von Authentifizierungsmechanismen. Wenn du sie aktivierst oder deaktivierst, werden sie auf dem Anmeldebildschirm angezeigt, damit Benutzer sie auswählen können. Klicke auf die Drei-Punkte, um den Authentifizierungsmechanismus zu konfigurieren.",
|
||||
"disabled": "Deaktiviert",
|
||||
"enabled": "Aktiviert",
|
||||
"enabledKey": "Aktiviert?",
|
||||
"oidc": "OpenID Connect",
|
||||
"simple": "Einfach (Nutzername/Passwort)",
|
||||
"srOpenOptions": "Einstellungen öffnen",
|
||||
"title": "Authentifizierung"
|
||||
},
|
||||
"authoptionsHeader": "Authentifizierungseinstellungen",
|
||||
"delete": "Löschen",
|
||||
"deleteUser": "Benutzer löschen {0}",
|
||||
"description": "Verwalte Benutzer auf deiner Drop-Instanz und konfiguriere deine Authentifizierungsmethode.",
|
||||
"displayNameHeader": "Anzeigename",
|
||||
"emailHeader": "E-Mail",
|
||||
"normalUserLabel": "Normaler Benutzer",
|
||||
"simple": {
|
||||
"adminInvitation": "Admin Einladung",
|
||||
"createInvitation": "Einladung erstellen",
|
||||
"description": "Die einfache Authentifizierung verwendet ein Einladungssystem zur Erstellung von Benutzern. Du kannst eine Einladung erstellen und optional einen Benutzernamen oder eine E-Mail-Adresse für den Benutzer angeben. Daraufhin wird eine magische URL generiert, mit der ein Konto erstellt werden kann.",
|
||||
"expires": "Läuft ab: {expiry}",
|
||||
"invitationTitle": "Einladungen",
|
||||
"invite3Days": "3 Tage",
|
||||
"invite6Months": "6 Monate",
|
||||
"inviteAdminSwitchDescription": "Erstelle diesen Benutzer als Administrator",
|
||||
"inviteAdminSwitchLabel": "Admin Einladung",
|
||||
"inviteButton": "Einladung",
|
||||
"inviteDescription": "Drop erstellt eine URL, die du an die Person senden kannst, die du einladen möchtest. Du kannst optional einen Benutzernamen oder eine E-Mail-Adresse angeben, die sie verwenden soll.",
|
||||
"inviteEmailDescription": "Muss im Format nutzer{'@'}beispiel.de sein",
|
||||
"inviteEmailLabel": "E-Mail-Adresse (optional)",
|
||||
"inviteEmailPlaceholder": "ich{'@'}beispiel.de",
|
||||
"inviteExpiryLabel": "Läuft ab",
|
||||
"inviteMonth": "1 Monat",
|
||||
"inviteNever": "Niemals",
|
||||
"inviteTitle": "Ein Benutzer zu Drop einladen",
|
||||
"inviteUsernameFormat": "Muss mindestens 5 Zeichen lang sein",
|
||||
"inviteUsernameLabel": "Nutzername (optional)",
|
||||
"inviteUsernamePlaceholder": "meinNutzername",
|
||||
"inviteWeek": "1 Woche",
|
||||
"inviteYear": "1 Jahr",
|
||||
"neverExpires": "Läuft niemals ab.",
|
||||
"noEmailEnforced": "Keine E-Mail erforderlich.",
|
||||
"noInvitations": "Keine Einladungen.",
|
||||
"noUsernameEnforced": "Kein Nutzername erforderlich.",
|
||||
"title": "Einfache Authentifizierung",
|
||||
"userInvitation": "Benutzereinladung"
|
||||
},
|
||||
"srEditLabel": "Bearbeiten",
|
||||
"usernameHeader": "Nutzername"
|
||||
}
|
||||
},
|
||||
"welcome": "Deutsche, willkommen!"
|
||||
}
|
||||
|
||||
@ -19,7 +19,7 @@
|
||||
"title": "Messages from the Crows' Nest",
|
||||
"unread": "Unread Messages"
|
||||
},
|
||||
"settings": "Account Settings, savvy?",
|
||||
"settings": "Settings",
|
||||
"title": "Yer Own Coffer"
|
||||
},
|
||||
"actions": "Deeds",
|
||||
@ -38,6 +38,10 @@
|
||||
"requestedAccess": "\"{name}\" has requested passage to yer Drop coffer.",
|
||||
"success": "Shiver me timbers, it worked!"
|
||||
},
|
||||
"code": {
|
||||
"description": "Use the secret map to dock ye ship when lacking a web surfer.",
|
||||
"title": "Dock ye ship"
|
||||
},
|
||||
"confirmPassword": "Confirm @:auth.password",
|
||||
"displayName": "Yer Scallywag Name",
|
||||
"email": "Salty Mail",
|
||||
@ -71,6 +75,7 @@
|
||||
"srComma": ", {0}"
|
||||
},
|
||||
"common": {
|
||||
"add": "Append",
|
||||
"cannotUndo": "This deed cannot be undone, ye hear!",
|
||||
"close": "Shut yer trap!",
|
||||
"create": "Forge!",
|
||||
@ -83,9 +88,12 @@
|
||||
"insert": "Insert",
|
||||
"name": "Name, argh!",
|
||||
"noResults": "No plunder found!",
|
||||
"noSelected": "No cargo selected.",
|
||||
"remove": "Walk the plank",
|
||||
"save": "Stow it!",
|
||||
"saved": "Preserved",
|
||||
"servers": "Ships",
|
||||
"srLoading": "Loading, loading, argh...",
|
||||
"srLoading": "Loading, loading, argh…",
|
||||
"tags": "Marks",
|
||||
"today": "Today"
|
||||
},
|
||||
@ -228,6 +236,8 @@
|
||||
"header": {
|
||||
"admin": {
|
||||
"admin": "Cap'n",
|
||||
"metadata": "Meta argh",
|
||||
"settings": "Shape",
|
||||
"tasks": "Duties",
|
||||
"users": "Crew"
|
||||
},
|
||||
@ -263,16 +273,18 @@
|
||||
},
|
||||
"gameLibrary": "Game Treasure Hoard",
|
||||
"import": {
|
||||
"bulkImportDescription": "When importing ye versions, ye won't be sent to the import duty.",
|
||||
"bulkImportTitle": "Plunder the imports",
|
||||
"import": "Import, ye dog!",
|
||||
"link": "Import {arrow}",
|
||||
"loading": "Loadin' plunder results, arrr...",
|
||||
"loading": "Loadin' plunder results, arrr…",
|
||||
"search": "Search",
|
||||
"searchPlaceholder": "Fallout 4, savvy?",
|
||||
"selectDir": "Pick a directory, ye landlubber...",
|
||||
"selectDir": "Pick a directory, ye landlubber…",
|
||||
"selectGame": "Pick plunder to import",
|
||||
"selectGamePlaceholder": "Pick a game, ye dog...",
|
||||
"selectGamePlaceholder": "Pick a game, ye dog…",
|
||||
"selectGameSearch": "Pick game",
|
||||
"selectPlatform": "Pick a ship, ye scallywag...",
|
||||
"selectPlatform": "Pick a ship, ye scallywag…",
|
||||
"version": {
|
||||
"advancedOptions": "Advanced options, savvy?",
|
||||
"import": "Import version",
|
||||
@ -280,7 +292,7 @@
|
||||
"launchCmd": "Launch executable/command, argh!",
|
||||
"launchDesc": "Executable to launch the game, matey!",
|
||||
"launchPlaceholder": "game.exe, aye!",
|
||||
"loadingVersion": "Loading version charts...",
|
||||
"loadingVersion": "Loading version charts…",
|
||||
"noAdv": "No advanced options for this rig, argh.",
|
||||
"noVersions": "No versions to import, savvy!",
|
||||
"platform": "Ship type",
|
||||
@ -298,6 +310,16 @@
|
||||
},
|
||||
"withoutMetadata": "Import without charts"
|
||||
},
|
||||
"metadata": {
|
||||
"companies": {
|
||||
"action": "Shape {arrow}",
|
||||
"addGame": {
|
||||
"developer": "Creator?",
|
||||
"noGames": "No games to plunder",
|
||||
"publisher": "Distributor?"
|
||||
}
|
||||
}
|
||||
},
|
||||
"metadataProvider": "Charts Provider",
|
||||
"noGames": "No plunder imported, savvy!",
|
||||
"openEditor": "Open in Editor {arrow}",
|
||||
@ -345,7 +367,7 @@
|
||||
"launcherOpen": "Open in Launcher, argh!",
|
||||
"noGames": "No plunder in treasure hoard, savvy!",
|
||||
"notFound": "Plunder not found, matey!",
|
||||
"search": "Search treasure hoard, ye dog...",
|
||||
"search": "Search treasure hoard, ye dog…",
|
||||
"subheader": "Sort yer plunder into collections for easy access, and get to all yer plunder, savvy!"
|
||||
},
|
||||
"lowest": "lowest",
|
||||
@ -360,7 +382,7 @@
|
||||
"preview": "Preview, matey!",
|
||||
"shortDesc": "Short description",
|
||||
"submit": "Submit, ye scurvy dog!",
|
||||
"tagPlaceholder": "Add a mark, ye dog...",
|
||||
"tagPlaceholder": "Add a mark, ye dog…",
|
||||
"titles": "Title, argh!",
|
||||
"uploadCover": "Hoist cover image"
|
||||
},
|
||||
@ -376,7 +398,7 @@
|
||||
"none": "No articles, savvy!",
|
||||
"notFound": "Article not found, matey!",
|
||||
"search": "Search articles, ye dog!",
|
||||
"searchPlaceholder": "Search articles, argh...",
|
||||
"searchPlaceholder": "Search articles, argh…",
|
||||
"subheader": "Stay up to date with the latest charts and announcements, savvy!",
|
||||
"title": "Latest News from the High Seas"
|
||||
},
|
||||
|
||||
@ -19,6 +19,28 @@
|
||||
"title": "Notifications",
|
||||
"unread": "Unread Notifications"
|
||||
},
|
||||
"token": {
|
||||
"title": "API Tokens",
|
||||
"subheader": "Manage your API tokens, and what they can access.",
|
||||
"name": "API token name",
|
||||
"nameDesc": "The name of the token, for reference.",
|
||||
"namePlaceholder": "My New Token",
|
||||
"acls": "ACLs/scopes",
|
||||
"aclsDesc": "Defines what this token has the authority to do. You should avoid selecting all ACLs, if they are not necessary.",
|
||||
"expiry": "Expiry",
|
||||
"noExpiry": "No expiry",
|
||||
"revoke": "Revoke",
|
||||
"noTokens": "No tokens connected to your account.",
|
||||
|
||||
"expiryMonth": "A month",
|
||||
"expiry3Month": "3 months",
|
||||
"expiry6Month": "6 months",
|
||||
"expiryYear": "A year",
|
||||
"expiry5Year": "5 years",
|
||||
|
||||
"success": "Successfully created token.",
|
||||
"successNote": "Make sure to copy it now, as it won't be shown again."
|
||||
},
|
||||
"settings": "Settings",
|
||||
"title": "Account Settings"
|
||||
},
|
||||
@ -39,8 +61,8 @@
|
||||
"success": "Successful!"
|
||||
},
|
||||
"code": {
|
||||
"title": "Connect your Drop client",
|
||||
"description": "Use a code to connect your Drop client if you are unable to open a web browser on your device."
|
||||
"description": "Use a code to connect your Drop client if you are unable to open a web browser on your device.",
|
||||
"title": "Connect your Drop client"
|
||||
},
|
||||
"confirmPassword": "Confirm @:auth.password",
|
||||
"displayName": "Display Name",
|
||||
@ -67,38 +89,6 @@
|
||||
"signout": "Signout",
|
||||
"username": "Username"
|
||||
},
|
||||
"setup": {
|
||||
"welcome": "Hey there.",
|
||||
"welcomeDescription": "Welcome to Drop setup wizard. It will walk you through configuring Drop for the first time, and how it works.",
|
||||
"finish": "Let's go {arrow}",
|
||||
"noPage": "no page",
|
||||
"auth": {
|
||||
"title": "Authentication",
|
||||
"description": "Authentication in Drop happens through multiple configured 'providers'. Each one can allow users to sign-in through their method. To get started, have at least one authentication provider enabled, and create an account through it.",
|
||||
"docs": "Documentation {arrow}",
|
||||
"enabled": "Enabled?",
|
||||
"simple": {
|
||||
"title": "Simple authentication",
|
||||
"description": "Simple authentication uses username/password to authentication users. It is enabled by default if no other authentication provider is enabled.",
|
||||
"register": "Register as admin {arrow}"
|
||||
},
|
||||
"openid": {
|
||||
"title": "OpenID Connect",
|
||||
"description": "OpenID Connect (OIDC) is an OAuth2 extension commonly supported. Drop requires OIDC configuration to be done via environment variables.",
|
||||
"skip": "I have a user with OIDC"
|
||||
}
|
||||
},
|
||||
"stages": {
|
||||
"account": {
|
||||
"name": "Setup your admin account.",
|
||||
"description": "You need at least one account to start using Drop."
|
||||
},
|
||||
"library": {
|
||||
"name": "Create a library.",
|
||||
"description": "Add at least one library source to use Drop."
|
||||
}
|
||||
}
|
||||
},
|
||||
"cancel": "Cancel",
|
||||
"chars": {
|
||||
"arrow": "→",
|
||||
@ -107,6 +97,7 @@
|
||||
"srComma": ", {0}"
|
||||
},
|
||||
"common": {
|
||||
"add": "Add",
|
||||
"cannotUndo": "This action cannot be undone.",
|
||||
"close": "Close",
|
||||
"create": "Create",
|
||||
@ -124,10 +115,9 @@
|
||||
"save": "Save",
|
||||
"saved": "Saved",
|
||||
"servers": "Servers",
|
||||
"srLoading": "Loading...",
|
||||
"srLoading": "Loading…",
|
||||
"tags": "Tags",
|
||||
"today": "Today",
|
||||
"add": "Add"
|
||||
"today": "Today"
|
||||
},
|
||||
"delete": "Delete",
|
||||
"drop": {
|
||||
@ -169,6 +159,10 @@
|
||||
"usernameTaken": "Username already taken."
|
||||
},
|
||||
"backHome": "{arrow} Back to home",
|
||||
"externalUrl": {
|
||||
"subtitle": "This message is only visible to admins.",
|
||||
"title": "Accessing over different EXTERNAL_URL. Please check the docs."
|
||||
},
|
||||
"game": {
|
||||
"banner": {
|
||||
"description": "Drop failed to update the banner image: {0}",
|
||||
@ -210,7 +204,7 @@
|
||||
},
|
||||
"source": {
|
||||
"delete": {
|
||||
"desc": "Drop couldn't add delete this source: {0}",
|
||||
"desc": "Drop couldn't delete this source: {0}",
|
||||
"title": "Failed to delete library source"
|
||||
}
|
||||
}
|
||||
@ -269,7 +263,11 @@
|
||||
"admin": {
|
||||
"admin": "Admin",
|
||||
"metadata": "Meta",
|
||||
"settings": "Settings",
|
||||
"settings": {
|
||||
"title": "Settings",
|
||||
"store": "Store",
|
||||
"tokens": "API tokens"
|
||||
},
|
||||
"tasks": "Tasks",
|
||||
"users": "Users"
|
||||
},
|
||||
@ -284,9 +282,7 @@
|
||||
"addToLib": "Add to Library",
|
||||
"admin": {
|
||||
"detectedGame": "Drop has detected you have new games to import.",
|
||||
"detectedVersion": "Drop has detected you have new verions of this game to import.",
|
||||
"offlineTitle": "Game offline",
|
||||
"offline": "Drop couldn't access this game.",
|
||||
"detectedVersion": "Drop has detected you have new versions of this game to import.",
|
||||
"game": {
|
||||
"addCarouselNoImages": "No images to add.",
|
||||
"addDescriptionNoImages": "No images to add.",
|
||||
@ -307,18 +303,18 @@
|
||||
},
|
||||
"gameLibrary": "Game Library",
|
||||
"import": {
|
||||
"bulkImportDescription": "When on this page, you won't be redirect to the import task, so you can import multiple games in succession.",
|
||||
"bulkImportTitle": "Bulk import mode",
|
||||
"import": "Import",
|
||||
"link": "Import {arrow}",
|
||||
"loading": "Loading game results...",
|
||||
"loading": "Loading game results…",
|
||||
"search": "Search",
|
||||
"searchPlaceholder": "Fallout 4",
|
||||
"selectDir": "Please select a directory...",
|
||||
"selectDir": "Please select a directory…",
|
||||
"selectGame": "Select game to import",
|
||||
"selectGamePlaceholder": "Please select a game...",
|
||||
"selectGamePlaceholder": "Please select a game…",
|
||||
"selectGameSearch": "Select game",
|
||||
"selectPlatform": "Please select a platform...",
|
||||
"bulkImportTitle": "Bulk import mode",
|
||||
"bulkImportDescription": "When on, this page won't redirect you to the import task, so you can import multiple games in succession.",
|
||||
"selectPlatform": "Please select a platform…",
|
||||
"version": {
|
||||
"advancedOptions": "Advanced options",
|
||||
"import": "Import version",
|
||||
@ -326,7 +322,7 @@
|
||||
"launchCmd": "Launch executable/command",
|
||||
"launchDesc": "Executable to launch the game",
|
||||
"launchPlaceholder": "game.exe",
|
||||
"loadingVersion": "Loading version metadata...",
|
||||
"loadingVersion": "Loading version metadata…",
|
||||
"noAdv": "No advanced options for this configuration.",
|
||||
"noVersions": "No versions to import",
|
||||
"platform": "Version platform",
|
||||
@ -344,21 +340,84 @@
|
||||
},
|
||||
"withoutMetadata": "Import without metadata"
|
||||
},
|
||||
"metadata": {
|
||||
"companies": {
|
||||
"action": "Manage {arrow}",
|
||||
"addGame": {
|
||||
"description": "Pick a game to add to the company, and whether it should be listed as a developer, publisher, or both.",
|
||||
"developer": "Developer?",
|
||||
"noGames": "No games to add",
|
||||
"publisher": "Publisher?",
|
||||
"title": "Connect game to this company"
|
||||
},
|
||||
"description": "Companies organize games by who they were developed or published by.",
|
||||
"editor": {
|
||||
"action": "Add Game {plus}",
|
||||
"descriptionPlaceholder": "{'<'}description{'>'}",
|
||||
"developed": "Developed",
|
||||
"libraryDescription": "Add, remove, or customise what this company has developed and/or published.",
|
||||
"libraryTitle": "Game Library",
|
||||
"noDescription": "(no description)",
|
||||
"published": "Published",
|
||||
"uploadBanner": "Upload banner",
|
||||
"uploadIcon": "Upload icon",
|
||||
"websitePlaceholder": "{'<'}website{'>'}"
|
||||
},
|
||||
"modals": {
|
||||
"createDescription": "Create a company to further organize your games.",
|
||||
"createFieldDescription": "Company Description",
|
||||
"createFieldDescriptionPlaceholder": "A small indie studio that...",
|
||||
"createFieldName": "Company Name",
|
||||
"createFieldNamePlaceholder": "My New Company...",
|
||||
"createFieldWebsite": "Company Website",
|
||||
"createFieldWebsitePlaceholder": "https://example.com/",
|
||||
"createTitle": "Create a company",
|
||||
"nameDescription": "Edit the company's name. Used to match to new game imports.",
|
||||
"nameTitle": "Edit company name",
|
||||
"shortDeckDescription": "Edit the company's description. Doesn't affect long (markdown) description.",
|
||||
"shortDeckTitle": "Edit company description",
|
||||
"websiteDescription": "Edit the company's website. Note: this will be a link, and won't have redirect protection.",
|
||||
"websiteTitle": "Edit company website"
|
||||
},
|
||||
"noCompanies": "No companies",
|
||||
"noGames": "No games",
|
||||
"search": "Search companies…",
|
||||
"searchGames": "Search company games…",
|
||||
"title": "Companies"
|
||||
},
|
||||
"tags": {
|
||||
"action": "Manage {arrow}",
|
||||
"create": "Create",
|
||||
"description": "Tags are automatically created from imported genres. You can add custom tags to add categorisation to your game library.",
|
||||
"modal": {
|
||||
"description": "Create a tag to organize your library.",
|
||||
"title": "Create Tag"
|
||||
},
|
||||
"title": "Tags"
|
||||
}
|
||||
},
|
||||
"metadataProvider": "Metadata provider",
|
||||
"noGames": "No games imported",
|
||||
"libraryHint": "No libraries configured.",
|
||||
"libraryHintDocsLink": "What does this mean? {arrow}",
|
||||
"offline": "Drop couldn't access this game.",
|
||||
"offlineTitle": "Game offline",
|
||||
"openEditor": "Open in Editor {arrow}",
|
||||
"openStore": "Open in Store",
|
||||
"shortDesc": "Short Description",
|
||||
"sources": {
|
||||
"create": "Create source",
|
||||
"edit": "Edit source",
|
||||
"createDesc": "Drop will use this source to access your game library, and make them available.",
|
||||
"desc": "Configure your library sources, where Drop will look for new games and versions to import.",
|
||||
"documentationLink": "Documentation {arrow}",
|
||||
"edit": "Edit source",
|
||||
"fsDesc": "Imports games from a path on disk. Requires version-based folder structure, and supports archived games.",
|
||||
"fsFlatDesc": "Imports games from a path on disk, but without a separate version subfolder. Useful when migrating an existing library to Drop.",
|
||||
"fsFlatTitle": "Compatibility",
|
||||
"fsPath": "Path",
|
||||
"fsPathDesc": "An absolute path to your game library.",
|
||||
"fsPathPlaceholder": "/mnt/games",
|
||||
"fsTitle": "Drop-style",
|
||||
"link": "Sources {arrow}",
|
||||
"nameDesc": "The name of your source, for reference.",
|
||||
"namePlaceholder": "My New Source",
|
||||
@ -373,53 +432,7 @@
|
||||
"noVersions": "You have no versions of this game available.",
|
||||
"noVersionsAdded": "no versions added"
|
||||
},
|
||||
"versionPriority": "Version priority",
|
||||
"metadata": {
|
||||
"tags": {
|
||||
"title": "Tags",
|
||||
"description": "Tags are automatically created from imported genres. You can add custom tags to add categorisation to your game library.",
|
||||
"action": "Manage {arrow}",
|
||||
"create": "Create",
|
||||
"modal": {
|
||||
"title": "Create Tag",
|
||||
"description": "Create a tag to organize your library."
|
||||
}
|
||||
},
|
||||
"companies": {
|
||||
"title": "Companies",
|
||||
"description": "Companies organize games by who they were developed or published by.",
|
||||
"action": "Manage {arrow}",
|
||||
"search": "Search companies...",
|
||||
"searchGames": "Search company games...",
|
||||
"noCompanies": "No companies",
|
||||
"noGames": "No games",
|
||||
"editor": {
|
||||
"libraryTitle": "Game Library",
|
||||
"libraryDescription": "Add, remove, or customise what this company has developed and/or published.",
|
||||
"action": "Add Game {plus}",
|
||||
"published": "Published",
|
||||
"developed": "Developed",
|
||||
"uploadIcon": "Upload icon",
|
||||
"uploadBanner": "Upload banner",
|
||||
"noDescription": "(no description)"
|
||||
},
|
||||
"addGame": {
|
||||
"title": "Connect game to this company",
|
||||
"description": "Pick a game to add to the company, and whether it should be listed as a developer, publisher, or both.",
|
||||
"publisher": "Publisher?",
|
||||
"developer": "Developer?",
|
||||
"noGames": "No games to add"
|
||||
},
|
||||
"modals": {
|
||||
"nameTitle": "Edit company name",
|
||||
"nameDescription": "Edit the company's name. Used to match to new game imports.",
|
||||
"shortDeckTitle": "Edit company description",
|
||||
"shortDeckDescription": "Edit the company's description. Doesn't affect long (markdown) description.",
|
||||
"websiteTitle": "Edit company website",
|
||||
"websiteDescription": "Edit the company's website. Note: this will be a link, and won't have redirect protection."
|
||||
}
|
||||
}
|
||||
}
|
||||
"versionPriority": "Version priority"
|
||||
},
|
||||
"back": "Back to Library",
|
||||
"collection": {
|
||||
@ -439,7 +452,7 @@
|
||||
"launcherOpen": "Open in Launcher",
|
||||
"noGames": "No games in library",
|
||||
"notFound": "Game not found",
|
||||
"search": "Search library...",
|
||||
"search": "Search library…",
|
||||
"subheader": "Organize your games into collections for easy access, and access all your games."
|
||||
},
|
||||
"lowest": "lowest",
|
||||
@ -454,7 +467,7 @@
|
||||
"preview": "Preview",
|
||||
"shortDesc": "Short description",
|
||||
"submit": "Submit",
|
||||
"tagPlaceholder": "Add a tag...",
|
||||
"tagPlaceholder": "Add a tag…",
|
||||
"titles": "Title",
|
||||
"uploadCover": "Upload cover image"
|
||||
},
|
||||
@ -470,7 +483,7 @@
|
||||
"none": "No articles",
|
||||
"notFound": "Article not found",
|
||||
"search": "Search articles",
|
||||
"searchPlaceholder": "Search articles...",
|
||||
"searchPlaceholder": "Search articles…",
|
||||
"subheader": "Stay up to date with the latest updates and announcements.",
|
||||
"title": "Latest News"
|
||||
},
|
||||
@ -490,6 +503,38 @@
|
||||
"title": "Settings"
|
||||
}
|
||||
},
|
||||
"setup": {
|
||||
"auth": {
|
||||
"description": "Authentication in Drop happens through multiple configured 'providers'. Each one can allow users to sign-in through their method. To get started, have at least one authentication provider enabled, and create an account through it.",
|
||||
"docs": "Documentation {arrow}",
|
||||
"enabled": "Enabled?",
|
||||
"openid": {
|
||||
"description": "OpenID Connect (OIDC) is an OAuth2 extension commonly supported. Drop requires OIDC configuration to be done via environment variables.",
|
||||
"skip": "I have a user with OIDC",
|
||||
"title": "OpenID Connect"
|
||||
},
|
||||
"simple": {
|
||||
"description": "Simple authentication uses username/password to authentication users. It is enabled by default if no other authentication provider is enabled.",
|
||||
"register": "Register as admin {arrow}",
|
||||
"title": "Simple authentication"
|
||||
},
|
||||
"title": "Authentication"
|
||||
},
|
||||
"finish": "Let's go {arrow}",
|
||||
"noPage": "no page",
|
||||
"stages": {
|
||||
"account": {
|
||||
"description": "You need at least one account to start using Drop.",
|
||||
"name": "Setup your admin account."
|
||||
},
|
||||
"library": {
|
||||
"description": "Add at least one library source to use Drop.",
|
||||
"name": "Create a library."
|
||||
}
|
||||
},
|
||||
"welcome": "Hey there.",
|
||||
"welcomeDescription": "Welcome to Drop setup wizard. It will walk you through configuring Drop for the first time, and how it works."
|
||||
},
|
||||
"store": {
|
||||
"about": "About",
|
||||
"commingSoon": "coming soon",
|
||||
@ -499,11 +544,13 @@
|
||||
"images": "Game Images",
|
||||
"lookAt": "Check it out",
|
||||
"noDevelopers": "No developers",
|
||||
"noGame": "no game",
|
||||
"noFeatured": "NO FEATURED GAMES",
|
||||
"noGame": "NO GAME",
|
||||
"noImages": "No images",
|
||||
"noPublishers": "No publishers.",
|
||||
"noTags": "No tags",
|
||||
"openAdminDashboard": "Open in Admin Dashboard",
|
||||
"openFeatured": "Star games in Admin Library {arrow}",
|
||||
"platform": "Platform | Platform | Platforms",
|
||||
"publishers": "Publishers | Publisher | Publishers",
|
||||
"rating": "Rating",
|
||||
@ -543,7 +590,9 @@
|
||||
"cleanupSessionsName": "Clean up sessions."
|
||||
},
|
||||
"viewTask": "View {arrow}",
|
||||
"weeklyScheduledTitle": "Weekly scheduled tasks"
|
||||
"weeklyScheduledTitle": "Weekly scheduled tasks",
|
||||
"progress": "{0}%",
|
||||
"execute": "{arrow} Execute"
|
||||
}
|
||||
},
|
||||
"title": "Drop",
|
||||
@ -592,7 +641,7 @@
|
||||
"createInvitation": "Create invitation",
|
||||
"description": "Simple authentication uses a system of 'invitations' to create users. You can create an invitation, and optionally specify a username or email for the user, and then it will generate a magic URL that can be used to create an account.",
|
||||
"expires": "Expires: {expiry}",
|
||||
"invitationTitle": "invitations",
|
||||
"invitationTitle": "Invitations",
|
||||
"invite3Days": "3 days",
|
||||
"invite6Months": "6 months",
|
||||
"inviteAdminSwitchDescription": "Create this user as an administrator",
|
||||
|
||||
@ -1 +1,639 @@
|
||||
{}
|
||||
{
|
||||
"account": {
|
||||
"devices": {
|
||||
"capabilities": "Capacités",
|
||||
"lastConnected": "Dernière Connexion",
|
||||
"noDevices": "Aucun appareil n'est connecté à vôtre compte.",
|
||||
"platform": "Plateforme",
|
||||
"revoke": "Révoquer",
|
||||
"subheader": "Gérer les appareils authorisés à accéder à votre compte Drop.",
|
||||
"title": "Appareils"
|
||||
},
|
||||
"notifications": {
|
||||
"all": "Tout voir {arrow}",
|
||||
"desc": "Voir et gérer vos notifications.",
|
||||
"markAllAsRead": "Tout marqué comme lu",
|
||||
"markAsRead": "Marquer comme lu",
|
||||
"none": "Pas de notification",
|
||||
"notifications": "Notifications",
|
||||
"title": "Notifications",
|
||||
"unread": "Notifications Non Lues"
|
||||
},
|
||||
"settings": "Paramètres",
|
||||
"title": "Paramètres du Compte",
|
||||
"token": {
|
||||
"acls": "ACLs/scopes",
|
||||
"aclsDesc": "Définir les permissions du Token. Il n'est pas recommandé de sélectionner toutes les ACLs, à moins que ce soit nécessaire.",
|
||||
"expiry": "Expiration",
|
||||
"expiry3Month": "3 mois",
|
||||
"expiry5Year": "5 Années",
|
||||
"expiry6Month": "6 mois",
|
||||
"expiryMonth": "Un mois",
|
||||
"expiryYear": "Une année",
|
||||
"name": "Nom du Token API",
|
||||
"nameDesc": "Le nom du Token, comme référence.",
|
||||
"namePlaceholder": "Mon nouveau Token",
|
||||
"noExpiry": "Pas d'expiration",
|
||||
"noTokens": "Aucun Token connecté à votre compte.",
|
||||
"revoke": "Révoquer",
|
||||
"subheader": "Gérer vos Tokens et leurs permissions associées.",
|
||||
"success": "Token créé avec succès.",
|
||||
"successNote": "Assurez vous de le sauvegarder maintenant, il ne sera plus disponible après.",
|
||||
"title": "API Tokens"
|
||||
}
|
||||
},
|
||||
"actions": "Actions",
|
||||
"add": "Ajouter",
|
||||
"adminTitle": "Tableau de Bord Administratif - Drop",
|
||||
"adminTitleTemplate": "{0} - Administration - Drop",
|
||||
"auth": {
|
||||
"callback": {
|
||||
"authClient": "Authoriser le client ?",
|
||||
"authorize": "Authoriser",
|
||||
"authorizedClient": "Drop a réussi a autoriser le client. Vous pouvez fermer cette fenêtre.",
|
||||
"issues": "Vous avez des problèmes ?",
|
||||
"learn": "En savoir plus {arrow}",
|
||||
"paste": "Collez ce code dans le client pour continuer :",
|
||||
"permWarning": "Accepter cette requête autorisera \"{name}\" sur \"{plateform} à :",
|
||||
"requestedAccess": "\"{name} a demandé accès à votre compte Drop.",
|
||||
"success": "Réussi !"
|
||||
},
|
||||
"code": {
|
||||
"description": "Utiliser un code pour vous connecter à votre client Drop si vous ne pouvez pas ouvrir un navigateur web sur votre appareil.",
|
||||
"title": "Connecter votre client Drop"
|
||||
},
|
||||
"displayName": "Nom d'Affichage",
|
||||
"email": "Email",
|
||||
"password": "Mot de passe",
|
||||
"register": {
|
||||
"confirmPasswordFormat": "Doit être pareil qu'au dessus",
|
||||
"emailFormat": "Doit être au format utilisateur{'@'}exemple.com",
|
||||
"passwordFormat": "Doit être au moins 14 caractères ou plus",
|
||||
"subheader": "Remplissez vos coordonnées pour créer votre compte.",
|
||||
"title": "Créer votre compte Drop",
|
||||
"usernameFormat": "Doit être au moins 5 caractères et en minuscules"
|
||||
},
|
||||
"signin": {
|
||||
"externalProvider": "Connectez vous avec un fournisseur externe {arrow}",
|
||||
"forgot": "Mot de passe oublié ?",
|
||||
"noAccount": "Pas de compte ? Demandez à un administrateur d'en créer un pour vous.",
|
||||
"or": "OU",
|
||||
"pageTitle": "Se connecter à Drop",
|
||||
"rememberMe": "Se souvenir de moi",
|
||||
"signin": "Se connecter",
|
||||
"title": "Se connecter à votre compte"
|
||||
},
|
||||
"signout": "Déconnexion",
|
||||
"username": "Nom d'utilisateur"
|
||||
},
|
||||
"cancel": "Annuler",
|
||||
"chars": {
|
||||
"arrow": "→",
|
||||
"arrowBack": "←",
|
||||
"quoted": "\"\"",
|
||||
"srComma": ", {0}"
|
||||
},
|
||||
"common": {
|
||||
"add": "Ajouter",
|
||||
"cannotUndo": "Cette action ne peut pas être défaite.",
|
||||
"close": "Fermer",
|
||||
"create": "Créer",
|
||||
"date": "Date",
|
||||
"deleteConfirm": "Êtes vous sûr de vouloir supprimer \"{0}\" ?",
|
||||
"divider": "{'|'}",
|
||||
"edit": "Éditer",
|
||||
"friends": "Amis",
|
||||
"groups": "Groupes",
|
||||
"insert": "Insérer",
|
||||
"name": "Nom",
|
||||
"noResults": "Pas de résultat",
|
||||
"noSelected": "Pas d'élément sélectionné.",
|
||||
"remove": "Retirer",
|
||||
"save": "Sauvegarder",
|
||||
"saved": "Sauvegardé",
|
||||
"servers": "Serveurs",
|
||||
"srLoading": "Chargement…",
|
||||
"tags": "Étiquettes",
|
||||
"today": "Aujourd'hui"
|
||||
},
|
||||
"delete": "Supprimer",
|
||||
"drop": {
|
||||
"desc": "Une plateforme de distribution libre conçue pour être rapide, flexible et belle.",
|
||||
"drop": "Drop"
|
||||
},
|
||||
"editor": {
|
||||
"bold": "Gras",
|
||||
"boldPlaceholder": "Caractères gras",
|
||||
"code": "Code",
|
||||
"codePlaceholder": "code",
|
||||
"heading": "En-tête",
|
||||
"headingPlaceholder": "en-tête",
|
||||
"italic": "Italique",
|
||||
"italicPlaceholder": "texte italique",
|
||||
"link": "Lien",
|
||||
"linkPlaceholder": "texte du lien",
|
||||
"listItem": "Élement de liste",
|
||||
"listItemPlaceholder": "élément de liste"
|
||||
},
|
||||
"errors": {
|
||||
"admin": {
|
||||
"user": {
|
||||
"delete": {
|
||||
"desc": "Drop n'a pas pu supprimer cet utilisateur : {0}",
|
||||
"title": "Échec de la suppression de l'utilisateur"
|
||||
}
|
||||
}
|
||||
},
|
||||
"auth": {
|
||||
"disabled": "Compte invalide or désactivé. Merci de contacter l'administrateur du serveur.",
|
||||
"invalidInvite": "Invitation invalide ou expirée",
|
||||
"invalidUserOrPass": "Nom d'utilisateur ou password invalide.",
|
||||
"inviteIdRequired": "id est requis pour récupérer l'invitation",
|
||||
"method": {
|
||||
"signinDisabled": "Méthode de connexion non activée"
|
||||
},
|
||||
"usernameTaken": "Nom d'utilisateur déjà pris."
|
||||
},
|
||||
"backHome": "{arrow} Retour a l'accueil",
|
||||
"game": {
|
||||
"banner": {
|
||||
"description": "Drop a échoué a mettre à jour l'image de la bannière : {0}",
|
||||
"title": "Échec de la mise à jour de l'image de la bannière"
|
||||
},
|
||||
"carousel": {
|
||||
"description": "Drop a échoué a mettre a jour le carrousel à images : {0}",
|
||||
"title": "Échec de la mise à jour du carrousel à images"
|
||||
},
|
||||
"cover": {
|
||||
"description": "Drop a échoué à mettre à jour l'image de couverture : {0}",
|
||||
"title": "Échec de la mise à jour de l'image de couverture"
|
||||
},
|
||||
"deleteImage": {
|
||||
"description": "Drop a échoué à supprimer l'image : {0}",
|
||||
"title": "Échec de la suppression de l'image"
|
||||
},
|
||||
"description": {
|
||||
"description": "Drop a échoué à mettre à jour la description du jeu : {0}",
|
||||
"title": "Échec de la mise à jour de la description du jeu"
|
||||
},
|
||||
"metadata": {
|
||||
"description": "Drop a échoué à mettre à jour les données méta : {0}",
|
||||
"title": "Échec de la mise à jour des données méta"
|
||||
}
|
||||
},
|
||||
"invalidBody": "Corps de requête non valide : {0}",
|
||||
"inviteRequired": "Invitation requise pour créer un compte.",
|
||||
"library": {
|
||||
"add": {
|
||||
"desc": "Drop n'a pas pu ajouter ce jeu à votre bibliothèque : {0}",
|
||||
"title": "Échec de l'ajout du jeu à la bibliothèque"
|
||||
},
|
||||
"collection": {
|
||||
"create": {
|
||||
"desc": "Drop n'a pas pu créer votre collection : {0}",
|
||||
"title": "Échec de la création de la collection"
|
||||
}
|
||||
},
|
||||
"source": {
|
||||
"delete": {
|
||||
"desc": "Drop n'a pas pu supprimer cette source : {0}",
|
||||
"title": "Échec de la suppression de la source de bibliothèque"
|
||||
}
|
||||
}
|
||||
},
|
||||
"news": {
|
||||
"article": {
|
||||
"delete": {
|
||||
"desc": "Drop n'a pas pu supprimer cet article : {0}",
|
||||
"title": "Échec de la suppression de l'article"
|
||||
}
|
||||
}
|
||||
},
|
||||
"occurred": "Une erreur s'est produite en réponse à vôtre requête. Si vous pensez que c'est un bug, merci de le rapporter. Essayer de vous connecter et voyez si cela résoud le problème.",
|
||||
"ohNo": "Oh non !",
|
||||
"pageTitle": "{0} | Drop",
|
||||
"revokeClient": "Échec de la révocation du client",
|
||||
"revokeClientFull": "Échec de la revocation du client {0}",
|
||||
"signIn": "Se connecter {arrow}",
|
||||
"unknown": "Une erreur inconnue est survenue",
|
||||
"upload": {
|
||||
"description": "Drop n'a pas pu uploader le fichier : {0}",
|
||||
"title": "Échec de l'upload du fichier"
|
||||
},
|
||||
"version": {
|
||||
"delete": {
|
||||
"desc": "Drop a rencontré une erreur pendant la suppression de la version : {error}",
|
||||
"title": "Une erreur est survenue pendant la supression de la version"
|
||||
},
|
||||
"order": {
|
||||
"desc": "Drop a rencontré une erreur pendant la mise a jour de la version : {error}",
|
||||
"title": "Une erreur est survenue pendant la mise a jour de l'ordre des versions"
|
||||
}
|
||||
}
|
||||
},
|
||||
"footer": {
|
||||
"about": "À propos",
|
||||
"aboutDrop": "À propos de Drop",
|
||||
"comparison": "Comparaison",
|
||||
"docs": {
|
||||
"client": "Documentation du client",
|
||||
"server": "Documentation du serveur"
|
||||
},
|
||||
"documentation": "Documentation",
|
||||
"findGame": "Trouver un jeu",
|
||||
"footer": "Pied de page",
|
||||
"games": "Jeux",
|
||||
"social": {
|
||||
"discord": "Discord",
|
||||
"github": "GitHub"
|
||||
},
|
||||
"topSellers": "Meilleures Ventes",
|
||||
"version": "Drop {version} {gitRef}"
|
||||
},
|
||||
"header": {
|
||||
"admin": {
|
||||
"admin": "Administration",
|
||||
"metadata": "Méta",
|
||||
"settings": "Paramètres",
|
||||
"tasks": "Tâches",
|
||||
"users": "Utilisateurs"
|
||||
},
|
||||
"back": "Retour",
|
||||
"openSidebar": "Ouvrir la barre latérale"
|
||||
},
|
||||
"helpUsTranslate": "Aidez nous à traduire Drop {arrow}",
|
||||
"highest": "le plus haut",
|
||||
"home": "Accueil",
|
||||
"library": {
|
||||
"addGames": "Tous les jeux",
|
||||
"addToLib": "Ajouter à la bibliothèque",
|
||||
"admin": {
|
||||
"detectedGame": "Drop a détecté que vous avez des nouveaux jeux a importer.",
|
||||
"detectedVersion": "Drop a détecté que vous avez des nouvelles versions de ce jeu à importer.",
|
||||
"game": {
|
||||
"addCarouselNoImages": "Pas d'image a ajouter.",
|
||||
"addDescriptionNoImages": "Pas d'image à ajouter.",
|
||||
"addImageCarousel": "Ajouter à partir d'une bibliothèque d'images",
|
||||
"currentBanner": "bannière",
|
||||
"currentCover": "couverture",
|
||||
"deleteImage": "Supprimer l'image",
|
||||
"editGameDescription": "Description du jeu",
|
||||
"editGameName": "Nom du jeu",
|
||||
"imageCarousel": "Carrousel d'images",
|
||||
"imageCarouselDescription": "Personnaliser quelles images et dans quel ordre elles sont affichées sur la page du Store.",
|
||||
"imageCarouselEmpty": "Aucune image n'a encore été ajoutée au carousel.",
|
||||
"imageLibrary": "Bibliothèque d'images",
|
||||
"imageLibraryDescription": "Veuillez noter que toutes les images uploadées sont accessible a tous les utilisateurs via des outils de développement des navigateurs.",
|
||||
"removeImageCarousel": "Retirer l'image",
|
||||
"setBanner": "Définir comme bannière",
|
||||
"setCover": "Définir comme couverture"
|
||||
},
|
||||
"gameLibrary": "Bibliothèque de jeux",
|
||||
"import": {
|
||||
"bulkImportDescription": "Lorsque vous êtes sur cette page, vous ne serez pas redirigé sur la tâche d'importation, pour que vous puissiez importer plusieurs jeux successivement.",
|
||||
"bulkImportTitle": "Mode d'importation de masse",
|
||||
"import": "Importer",
|
||||
"link": "Imported {arrow}",
|
||||
"loading": "Chargement des résultats des jeux…",
|
||||
"search": "Rechercher",
|
||||
"searchPlaceholder": "Fallout 4",
|
||||
"selectDir": "Merci de choisir un dossier…",
|
||||
"selectGame": "Sélectionnez le jeu à importer",
|
||||
"selectGamePlaceholder": "Merci de sélectionner un jeu…",
|
||||
"selectGameSearch": "Sélectionner un jeu",
|
||||
"selectPlatform": "Merci de sélectionner une plateforme…",
|
||||
"version": {
|
||||
"advancedOptions": "Options avancées",
|
||||
"import": "Importer une version",
|
||||
"installDir": "(install_dir)/",
|
||||
"launchCmd": "Lancer l'exécutable/commande",
|
||||
"launchDesc": "Exécutable pour lancer le jeu",
|
||||
"launchPlaceholder": "jeu.exe",
|
||||
"loadingVersion": "Chargement des métadonnées de la version…",
|
||||
"noAdv": "Pas d'option avancée pour cette configuration.",
|
||||
"noVersions": "Pas de version à importer",
|
||||
"platform": "Version de la plateforme",
|
||||
"setupCmd": "Exécutable/commande d'installation",
|
||||
"setupDesc": "Exécuté une fois lorsque le jeu a été installé",
|
||||
"setupMode": "Mode de configuration",
|
||||
"setupModeDesc": "Lorsqu'elle est activée, cette version n'a pas de commande de lancement, et exécute simplement l'exécutable sur l'ordinateur de l'utilisateur. Utile pour les jeux qui distribue uniquement des fichiers d'installation et non les fichiers portables.",
|
||||
"setupPlaceholder": "setup.exe",
|
||||
"umuLauncherId": "UMU Launcher ID",
|
||||
"umuOverride": "Remplacer l'ID de jeu du lanceur UMU",
|
||||
"umuOverrideDesc": "Par défaut, Drop utilise un non-ID pour lancer les jeux avec UMU Launcher. Pour récupérer les bons patchs pour certains jeux, vous pourriez avoir besoin de changer ce champ manuellement.",
|
||||
"updateMode": "Mode de mise à jour",
|
||||
"updateModeDesc": "Lorsqu'ils sont activés, ces fichiers seront installés par-dessus (remplaçant) la version précédente. Si plusieurs \"modes de mise à jour\" sont enchaînés, ils sont appliqués dans l'ordre.",
|
||||
"version": "Sélectionner la version à importer"
|
||||
},
|
||||
"withoutMetadata": "Importer sans les données méta"
|
||||
},
|
||||
"metadata": {
|
||||
"companies": {
|
||||
"action": "Gérer {arrow}",
|
||||
"addGame": {
|
||||
"description": "Choisissez un jeu à ajouter à la société, et si il faudrait la lister en tant que développeur, éditeur, ou les deux.",
|
||||
"developer": "Développeur ?",
|
||||
"noGames": "Pas de jeu à ajouter",
|
||||
"publisher": "Éditeur ?",
|
||||
"title": "Connecter le jeu a cette société"
|
||||
},
|
||||
"description": "Les sociétés organisent les jeux par qui les a développer ou éditer.",
|
||||
"editor": {
|
||||
"action": "Ajouter un jeu {plus}",
|
||||
"developed": "Développé",
|
||||
"libraryDescription": "Ajouter, supprimer ou personnaliser ce que cette société a développé et/ou publié.",
|
||||
"libraryTitle": "Bibliothèque de jeux",
|
||||
"noDescription": "(pas de description)",
|
||||
"published": "Publié",
|
||||
"uploadBanner": "Uploader bannière",
|
||||
"uploadIcon": "Uplader icône"
|
||||
},
|
||||
"modals": {
|
||||
"nameDescription": "Éditer le nom de la société. Ce nom est utilisé pour trouver les jeux nouvellement importés.",
|
||||
"nameTitle": "Éditer le nom de la société",
|
||||
"shortDeckDescription": "Éditer la description de la company. Cela n'affecte pas la description longue (markdown).",
|
||||
"shortDeckTitle": "Éditer la description de la société",
|
||||
"websiteDescription": "Éditer le site internet de la société. Note : cela sera un lien, et ne bénéficiera pas de la protection aux redirects.",
|
||||
"websiteTitle": "Éditer le site internet de la société"
|
||||
},
|
||||
"noCompanies": "Pas de société",
|
||||
"noGames": "Pas de jeu",
|
||||
"search": "Chercher des sociétés…",
|
||||
"searchGames": "Chercher les jeux de l'entreprise…",
|
||||
"title": "Sociétés"
|
||||
},
|
||||
"tags": {
|
||||
"action": "Gérer {arrow}",
|
||||
"create": "Créer",
|
||||
"description": "Les tags sont automatiquement créés à partir des genres importés. Vous pouvez ajouter des tags personnalisés pour ajouter la catégorisation de votre bibliothèque de jeux.",
|
||||
"modal": {
|
||||
"description": "Créer un tag pour organiser votre bibliothèque.",
|
||||
"title": "Créer un tag"
|
||||
},
|
||||
"title": "Tags"
|
||||
}
|
||||
},
|
||||
"metadataProvider": "Fournisseur de données méta",
|
||||
"noGames": "Pas de jeu importé",
|
||||
"offline": "Drop n'a pas pu accéder à ce jeu.",
|
||||
"offlineTitle": "Jeu hors-ligne",
|
||||
"openEditor": "Ouvrir dans l'éditeur {arrow}",
|
||||
"openStore": "Ouvrir dans le Store",
|
||||
"shortDesc": "Description Courte",
|
||||
"sources": {
|
||||
"create": "Créer une source",
|
||||
"createDesc": "Drop va utiliser cette source pour accéder à votre bibliothèque de jeux, et les rendre disponible.",
|
||||
"desc": "Configurer vos sources de bibliothèques où Drop va regarder pour les nouveaux jeux et versions à importer.",
|
||||
"edit": "Éditer la source",
|
||||
"fsDesc": "Importe les jeux à partir d'un chemin d'accès sur le disque. Cela requière une structure des dossiers basées sur la version, et qui supporte les jeux archivés.",
|
||||
"fsFlatDesc": "Importe les jeux à partir d'un chemin d’accès sur le disque, mais sans le sous-dossier version séparé. Utile pour migrer une bibliothèque vers Drop.",
|
||||
"fsPath": "Chemin d’accès",
|
||||
"fsPathDesc": "Un chemin d’accès absolu à votre bibliothèque de jeux.",
|
||||
"fsPathPlaceholder": "/mnt/jeux",
|
||||
"link": "Sources {arrow}",
|
||||
"nameDesc": "Le nom de votre source, pour référence.",
|
||||
"namePlaceholder": "Mes Nouvelle Source",
|
||||
"sources": "Sources de Bibliothèques",
|
||||
"typeDesc": "Le type de source. Affecte les options requises.",
|
||||
"working": "Marche ?"
|
||||
},
|
||||
"subheader": "Lorsque que vous rajoutez des dossiers à vos sources de bibliothèques, Drop le détectera et vous demandera de les importer. Chaque jeu a besoin d’être importé avant que vous puissiez importer une version.",
|
||||
"title": "Bibliothèques",
|
||||
"version": {
|
||||
"delta": "Mode de mise à jour",
|
||||
"noVersions": "Vous n'avez aucune version de ce jeu de disponible.",
|
||||
"noVersionsAdded": "pas de version ajoutée"
|
||||
},
|
||||
"versionPriority": "Priorité des versions"
|
||||
},
|
||||
"back": "Retour à la Bibliothèque",
|
||||
"collection": {
|
||||
"addToNew": "Ajouter à une nouvelle collection",
|
||||
"collections": "Collections",
|
||||
"create": "Créer une Collection",
|
||||
"createDesc": "Les collections peuvent être utilisées pour organiser vos jeux et vous permettre de les trouver plus facilement, surtout si vous possédez une grosse bibliothèque.",
|
||||
"delete": "Supprimer la Collection",
|
||||
"namePlaceholder": "Nom de la collection",
|
||||
"noCollections": "Pas de collection",
|
||||
"notFound": "Collection non trouvée",
|
||||
"subheader": "Ajouter une nouvelle collection pour organiser vos jeux",
|
||||
"title": "Collection"
|
||||
},
|
||||
"gameCount": "{0} jeux | {0} jeu | {0} jeux",
|
||||
"inLib": "Dans la Bibliothèque",
|
||||
"launcherOpen": "Ouvrir dans le Launcher",
|
||||
"noGames": "Pas de jeu dans la bibliothèque",
|
||||
"notFound": "Jeu non trouvé",
|
||||
"search": "Chercher bibliothèque…",
|
||||
"subheader": "Organiser vos jeux en collections pour un accès facile, et accéder à tous vos jeux."
|
||||
},
|
||||
"lowest": "le plus bas",
|
||||
"news": {
|
||||
"article": {
|
||||
"add": "Ajouter",
|
||||
"content": "Contenu (Markdown)",
|
||||
"create": "Créer un Nouvel Article",
|
||||
"editor": "Éditeur",
|
||||
"editorGuide": "Utilisez les raccourcis ci-dessus ou écrivez en Markdown directement. Supporte **gras**, *italique*, [lines](adresse), et plus.",
|
||||
"new": "Nouvel article",
|
||||
"preview": "Aperçu",
|
||||
"shortDesc": "Description courte",
|
||||
"submit": "Soumettre",
|
||||
"tagPlaceholder": "Ajouter un tag…",
|
||||
"titles": "Titre",
|
||||
"uploadCover": "Uploader l'image de couverture"
|
||||
},
|
||||
"back": "Retour aux Nouvelles",
|
||||
"checkLater": "Vérifier plus tard pour les mises à jour.",
|
||||
"delete": "Supprimer l'Article",
|
||||
"filter": {
|
||||
"month": "Ce mois",
|
||||
"week": "Cette semaine",
|
||||
"year": "Cette année"
|
||||
},
|
||||
"none": "Pas d'article",
|
||||
"notFound": "Article non trouvé",
|
||||
"search": "Chercher des articles",
|
||||
"searchPlaceholder": "Chercher des articles…",
|
||||
"subheader": "Rester à jour avec les dernières mises à et annonces.",
|
||||
"title": "Dernières Nouvelles"
|
||||
},
|
||||
"options": "Options",
|
||||
"security": "Sécurité",
|
||||
"selectLanguage": "Sélectionner la langue",
|
||||
"settings": {
|
||||
"admin": {
|
||||
"description": "Configurer les paramètres de Drop",
|
||||
"store": {
|
||||
"dropGameAltPlaceholder": "Exemple d'icône de Jeu",
|
||||
"dropGameDescriptionPlaceholder": "Ceci est un jeu exemple. Il sera remplacé si vous importez un jeu.",
|
||||
"dropGameNamePlaceholder": "Jeu Exemple",
|
||||
"showGamePanelTextDecoration": "Afficher le titre et la description sur les tuiles de jeu (par défaut : activé)",
|
||||
"title": "Store"
|
||||
},
|
||||
"title": "Paramètres"
|
||||
}
|
||||
},
|
||||
"setup": {
|
||||
"auth": {
|
||||
"description": "Authentification sur Drop se passe à travers multiple 'fournisseurs' pré-configuré. Chaque fournisseur peut autoriser des utilisateurs à se connecter via leurs méthodes. Pour commencer, aillez au moins un fournisseur d'authentification d'activé, et créer un compte via celui-ci.",
|
||||
"docs": "Documentation {arrow}",
|
||||
"enabled": "Activé ?",
|
||||
"openid": {
|
||||
"description": "OpenID Connect (OIDC) est une extension OAuth2 communément supportée. Drop requière que la configuration OIDC se fasse via des variables d'environnement.",
|
||||
"skip": "J'ai un utiliser avec OIDC",
|
||||
"title": "OpenID Connect"
|
||||
},
|
||||
"simple": {
|
||||
"description": "L'authentification simple utilise le nom d'utiliser et mot de passe pour authentifier les utilisateurs. Elle est activée par défaut si aucun autre fournisseur d'authentification est activé.",
|
||||
"register": "Créer un compte administrateur",
|
||||
"title": "Authentification simple"
|
||||
},
|
||||
"title": "Authentification"
|
||||
},
|
||||
"finish": "C'est parti {arrow}",
|
||||
"noPage": "Pas de page",
|
||||
"stages": {
|
||||
"account": {
|
||||
"description": "Vous avez besoin d'au moins un compte pour démarrer Drop.",
|
||||
"name": "Configurez votre compte administrateur."
|
||||
},
|
||||
"library": {
|
||||
"description": "Ajouter au moins une source de bibliothèques pour utiliser Drop.",
|
||||
"name": "Créer une bibliothèque."
|
||||
}
|
||||
},
|
||||
"welcome": "Salut.",
|
||||
"welcomeDescription": "Bienvenue dans l'assistant de configuration de Drop. Il va aidera configurer Drop pour la première fois, et vous expliquera son fonctionnement."
|
||||
},
|
||||
"store": {
|
||||
"about": "À propos",
|
||||
"commingSoon": "prochainement",
|
||||
"exploreMore": "Explorer plus {arrow}",
|
||||
"featured": "Mis en avant",
|
||||
"images": "Images de Jeux",
|
||||
"noDevelopers": "Pas de développeur",
|
||||
"noGame": "pas de jeu",
|
||||
"noImages": "Pas d'image",
|
||||
"noPublishers": "Pas d'éditeur.",
|
||||
"noTags": "Pas de tag",
|
||||
"openAdminDashboard": "Ouvrir dans le Tableau de Bord d'Administration",
|
||||
"platform": "Plateforme | Plateforme | Plateformes",
|
||||
"publishers": "Éditeurs | Éditeur | Éditeurs",
|
||||
"rating": "Note",
|
||||
"readLess": "Cliquez pour lire moins",
|
||||
"readMore": "Clique pour lire plus",
|
||||
"recentlyAdded": "Ajouté Récemment",
|
||||
"recentlyReleased": "Récemment publié",
|
||||
"recentlyUpdated": "Récemment Mis à Jour",
|
||||
"released": "Publié",
|
||||
"reviews": "({0} Avis)",
|
||||
"tags": "Tags",
|
||||
"title": "Store",
|
||||
"view": {
|
||||
"sort": "Trier",
|
||||
"srFilters": "Filtres",
|
||||
"srGames": "Jeux",
|
||||
"srViewGrid": "Voir grille"
|
||||
},
|
||||
"viewInStore": "Voir dans le Store",
|
||||
"website": "Site internet"
|
||||
},
|
||||
"tasks": {
|
||||
"admin": {
|
||||
"back": "{arrow} Retour aux Tâches",
|
||||
"completedTasksTitle": "Tâches complétées",
|
||||
"dailyScheduledTitle": "Tâches quotidiennes planifiées",
|
||||
"noTasksRunning": "Pas de tâche en cours",
|
||||
"runningTasksTitle": "Tâches en cours d'exécution",
|
||||
"scheduled": {
|
||||
"checkUpdateDescription": "Vérifier si Drop a une mise à jour.",
|
||||
"checkUpdateName": "Vérifier la mise à jour.",
|
||||
"cleanupInvitationsDescription": "Nettoie les invitations expirées de la base de données pour économiser de l'espace.",
|
||||
"cleanupInvitationsName": "Nettoie les invitations",
|
||||
"cleanupObjectsDescription": "Détecte et supprime les objets non référencés et non utilisés pour économiser de l'espace.",
|
||||
"cleanupObjectsName": "Nettoyer les objets",
|
||||
"cleanupSessionsDescription": "Nettoie les sessions expirées pour économiser de l'espace et assurer la sécurité.",
|
||||
"cleanupSessionsName": "Nettoie les sessions."
|
||||
},
|
||||
"viewTask": "Voir {arrow}",
|
||||
"weeklyScheduledTitle": "Tâches hebdomadaires planifiées"
|
||||
}
|
||||
},
|
||||
"title": "Drop",
|
||||
"titleTemplate": "{0} - Drop",
|
||||
"todo": "À faire",
|
||||
"type": "Type",
|
||||
"upload": "Uploader",
|
||||
"uploadFile": "Uploader fichier",
|
||||
"userHeader": {
|
||||
"closeSidebar": "Fermer la barre latérale",
|
||||
"links": {
|
||||
"community": "Communauté",
|
||||
"library": "Bibliothèque",
|
||||
"news": "Nouvelles"
|
||||
},
|
||||
"profile": {
|
||||
"admin": "Tableau de Bord Administratif",
|
||||
"settings": "Paramètres du compte"
|
||||
}
|
||||
},
|
||||
"users": {
|
||||
"admin": {
|
||||
"adminHeader": "Administrateur ?",
|
||||
"adminUserLabel": "Administrateur",
|
||||
"authLink": "Authentification {arrow}",
|
||||
"authentication": {
|
||||
"configure": "Configurer",
|
||||
"description": "Drop supporte une variété de \"mécanismes d'authentification\". Lorsque vous les activez ou les désactivez, ils sont affichés sur la page de connection pour que les utilisateurs puissent les sélectionner. Cliquer sur le menu à points pour configurer le mécanisme d'authentification.",
|
||||
"disabled": "Désactivé",
|
||||
"enabled": "Activé",
|
||||
"oidc": "OpenID Connect",
|
||||
"simple": "Simple (nom d'utilisateur/mot de passe)",
|
||||
"srOpenOptions": "Ouvrir les options",
|
||||
"title": "Authentification"
|
||||
},
|
||||
"authoptionsHeader": "Options Auth",
|
||||
"delete": "Supprimer",
|
||||
"deleteUser": "Supprimer l'utilisateur {0}",
|
||||
"description": "Gérer les utilisateurs sur votre instance Drop, et configurer vos méthodes d'authentification.",
|
||||
"displayNameHeader": "Nom d'affichage",
|
||||
"emailHeader": "Email",
|
||||
"normalUserLabel": "Utilisateur normal",
|
||||
"simple": {
|
||||
"adminInvitation": "Invitation adminstrateur",
|
||||
"createInvitation": "Créer invitation",
|
||||
"description": "L'authentification simple utilise un système d'invitations pour créer les utilisateurs. Tu peux créer une invitation et optionnellement spécifier le nom d'utilisateur ou email de cet utilisateur, et un lien magique sera généré un lien magique qui peut être utilisé pour créer le compte.",
|
||||
"expires": "Expire : {expiry}",
|
||||
"invitationTitle": "invitations",
|
||||
"invite3Days": "3 jours",
|
||||
"invite6Months": "6 mois",
|
||||
"inviteAdminSwitchDescription": "Créer cet utilisateur en tant qu'adminstrateur",
|
||||
"inviteAdminSwitchLabel": "Invitation adminstrateur",
|
||||
"inviteButton": "Invitation",
|
||||
"inviteDescription": "Drop va générer une adresse que vous pouvez envoyer à la personne que vous voulez inviter. Vous pouvez optionnellement spécifier un nom d'utilisateur ou une adresse e-mail qu'elle pourra utiliser.",
|
||||
"inviteEmailDescription": "Doit être dans le format utilisateur{'@'}exemple.com",
|
||||
"inviteEmailLabel": "E-mail adresse (optionnel)",
|
||||
"inviteEmailPlaceholder": "moi{'@'}exemple.com",
|
||||
"inviteExpiryLabel": "Expire",
|
||||
"inviteMonth": "1 mois",
|
||||
"inviteNever": "Jamais",
|
||||
"inviteTitle": "Inviter l'utilisateur sur Drop",
|
||||
"inviteUsernameFormat": "Doit être 5 caractères ou plus",
|
||||
"inviteUsernameLabel": "Nom d'utilisateur (optionnel)",
|
||||
"inviteUsernamePlaceholder": "monNomDUtilisateur",
|
||||
"inviteWeek": "1 semaine",
|
||||
"inviteYear": "1 an",
|
||||
"neverExpires": "N'expire jamais.",
|
||||
"noEmailEnforced": "Pas d'e-mail imposé.",
|
||||
"noInvitations": "Pas d'invitation.",
|
||||
"noUsernameEnforced": "Pas de nom d'utilisateur imposé.",
|
||||
"title": "Authentication simple",
|
||||
"userInvitation": "Invitation utilisateur"
|
||||
},
|
||||
"srEditLabel": "Éditer",
|
||||
"usernameHeader": "Nom d'utilisateur"
|
||||
}
|
||||
},
|
||||
"welcome": "Américain, bienvenue !"
|
||||
}
|
||||
|
||||
104
i18n/locales/ru.json
Normal file
104
i18n/locales/ru.json
Normal file
@ -0,0 +1,104 @@
|
||||
{
|
||||
"account": {
|
||||
"devices": {
|
||||
"capabilities": "Возможности",
|
||||
"lastConnected": "Последнее подключение",
|
||||
"noDevices": "К вашей учетной записи не подключено ни одного устройства.",
|
||||
"platform": "Платформа",
|
||||
"revoke": "Аннулировать",
|
||||
"subheader": "Управляйте устройствами, имеющими доступ к вашей учетной записи Drop.",
|
||||
"title": "Устройства"
|
||||
},
|
||||
"notifications": {
|
||||
"all": "Показать все {arrow}",
|
||||
"desc": "Просмотр и управление уведомлениями.",
|
||||
"markAllAsRead": "Отметить все как прочитанные",
|
||||
"markAsRead": "Отметить как прочитанное",
|
||||
"none": "Нет уведомлений",
|
||||
"notifications": "Уведомления",
|
||||
"title": "Уведомления",
|
||||
"unread": "Непрочитанные уведомления"
|
||||
},
|
||||
"settings": "Настройки",
|
||||
"title": "Настройки учетной записи",
|
||||
"token": {
|
||||
"acls": "Доступ и права",
|
||||
"aclsDesc": "Определяет, какие действия разрешены для этого токена. Не выбирайте все ACL, если это не требуется.",
|
||||
"expiry": "Истечение срока",
|
||||
"expiry3Month": "3 месяца",
|
||||
"expiry5Year": "5 лет",
|
||||
"expiry6Month": "6 месяцев",
|
||||
"expiryMonth": "Месяц",
|
||||
"expiryYear": "Год",
|
||||
"name": "Название API-токена",
|
||||
"nameDesc": "Название токена для справки.",
|
||||
"namePlaceholder": "Мои новые токены",
|
||||
"noExpiry": "Без срока действия",
|
||||
"noTokens": "На вашем аккаунте нет подключённых токенов.",
|
||||
"revoke": "Аннулировать",
|
||||
"subheader": "Управляйте своими API-токенами и их доступом.",
|
||||
"success": "Токен успешно создан.",
|
||||
"successNote": "Скопируйте токен сейчас, позже его не будет видно.",
|
||||
"title": "API-Токены"
|
||||
}
|
||||
},
|
||||
"actions": "Действия",
|
||||
"add": "Добавить",
|
||||
"adminTitle": "Панель администратора - Drop",
|
||||
"adminTitleTemplate": "{0} - Админ - Drop",
|
||||
"auth": {
|
||||
"callback": {
|
||||
"authClient": "Авторизовать клиента?",
|
||||
"authorize": "Авторизовать",
|
||||
"authorizedClient": "Drop успешно авторизовал клиента. Теперь вы можете закрыть это окно.",
|
||||
"issues": "Есть проблемы?",
|
||||
"learn": "Узнать больше {arrow}",
|
||||
"paste": "Вставьте этот код в клиент, чтобы продолжить:",
|
||||
"permWarning": "Принятие этого запроса позволит \"{name}\" на \"{platform}\" выполнять следующие действия:",
|
||||
"requestedAccess": "\"{name}\" запросил доступ к вашей учетной записи Drop.",
|
||||
"success": "Успешно!"
|
||||
},
|
||||
"email": "Элетронная почка",
|
||||
"password": "Пароль",
|
||||
"register": {
|
||||
"confirmPasswordFormat": "Должно совпадать с выше."
|
||||
},
|
||||
"signin": {
|
||||
"forgot": "Забыли пароль?",
|
||||
"noAccount": "Нет аккаунта? Попросите администратора создать его для вас.",
|
||||
"or": "Или",
|
||||
"signin": "Войти",
|
||||
"title": "Войдите в свой аккаунт"
|
||||
},
|
||||
"signout": "Выход",
|
||||
"username": "Имя пользователя"
|
||||
},
|
||||
"cancel": "Отмена",
|
||||
"common": {
|
||||
"add": "Добавить",
|
||||
"close": "Закрыть",
|
||||
"create": "Создать",
|
||||
"date": "Дата",
|
||||
"deleteConfirm": "Вы точно хотите удалить \"{0}\"?",
|
||||
"edit": "Редактировать",
|
||||
"friends": "Друзья",
|
||||
"groups": "Группы",
|
||||
"name": "Имя",
|
||||
"noResults": "Нет результатов",
|
||||
"noSelected": "Не выбранные предметы.",
|
||||
"remove": "Удалить",
|
||||
"save": "Сохранить",
|
||||
"saved": "Сохранено",
|
||||
"servers": "Сервера",
|
||||
"srLoading": "Загрузка…",
|
||||
"tags": "Теги",
|
||||
"today": "Сегодня"
|
||||
},
|
||||
"delete": "Удалить",
|
||||
"drop": {
|
||||
"drop": "Уронить"
|
||||
},
|
||||
"editor": {
|
||||
"link": "Ссылка"
|
||||
}
|
||||
}
|
||||
@ -200,7 +200,7 @@ const navigation: Array<NavigationItem & { icon: Component }> = [
|
||||
icon: RectangleStackIcon,
|
||||
},
|
||||
{
|
||||
label: $t("header.admin.settings"),
|
||||
label: $t("header.admin.settings.title"),
|
||||
route: "/admin/settings",
|
||||
prefix: "/admin/settings",
|
||||
icon: Cog6ToothIcon,
|
||||
|
||||
@ -74,7 +74,8 @@ export default defineNuxtConfig({
|
||||
|
||||
vite: {
|
||||
plugins: [
|
||||
tailwindcss(),
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
tailwindcss() as any,
|
||||
// only used in dev server, not build because nitro sucks
|
||||
// see build hook below
|
||||
viteStaticCopy({
|
||||
@ -84,7 +85,8 @@ export default defineNuxtConfig({
|
||||
dest: "twemoji",
|
||||
},
|
||||
],
|
||||
}),
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
}) as any,
|
||||
],
|
||||
},
|
||||
|
||||
@ -138,6 +140,7 @@ export default defineNuxtConfig({
|
||||
|
||||
scheduledTasks: {
|
||||
"0 * * * *": ["dailyTasks"],
|
||||
"*/30 * * * *": ["downloadCleanup"],
|
||||
},
|
||||
|
||||
storage: {
|
||||
@ -253,6 +256,7 @@ export default defineNuxtConfig({
|
||||
"https://www.giantbomb.com",
|
||||
"https://images.pcgamingwiki.com",
|
||||
"https://images.igdb.com",
|
||||
"https://*.steamstatic.com",
|
||||
],
|
||||
},
|
||||
strictTransportSecurity: false,
|
||||
|
||||
16
package.json
16
package.json
@ -1,9 +1,12 @@
|
||||
{
|
||||
"name": "drop",
|
||||
"version": "0.3.1",
|
||||
"version": "0.3.3",
|
||||
"private": true,
|
||||
"type": "module",
|
||||
"license": "AGPL-3.0-or-later",
|
||||
"engines": {
|
||||
"node": ">=22.16.0"
|
||||
},
|
||||
"scripts": {
|
||||
"build": "nuxt build",
|
||||
"dev": "nuxt dev",
|
||||
@ -11,14 +14,14 @@
|
||||
"preview": "nuxt preview",
|
||||
"postinstall": "nuxt prepare && prisma generate",
|
||||
"typecheck": "nuxt typecheck",
|
||||
"lint": "yarn lint:eslint && yarn lint:prettier",
|
||||
"lint": "pnpm run lint:eslint && pnpm run lint:prettier",
|
||||
"lint:eslint": "eslint .",
|
||||
"lint:prettier": "prettier . --check",
|
||||
"lint:fix": "eslint . --fix && prettier --write --list-different ."
|
||||
},
|
||||
"dependencies": {
|
||||
"@discordapp/twemoji": "^16.0.1",
|
||||
"@drop-oss/droplet": "1.6.0",
|
||||
"@drop-oss/droplet": "3.2.0",
|
||||
"@headlessui/vue": "^1.7.23",
|
||||
"@heroicons/vue": "^2.1.5",
|
||||
"@lobomfz/prismark": "0.0.3",
|
||||
@ -30,7 +33,7 @@
|
||||
"@vueuse/nuxt": "13.6.0",
|
||||
"argon2": "^0.43.0",
|
||||
"arktype": "^2.1.10",
|
||||
"axios": "^1.7.7",
|
||||
"axios": "^1.12.0",
|
||||
"bcryptjs": "^3.0.2",
|
||||
"cheerio": "^1.0.0",
|
||||
"cookie-es": "^2.0.0",
|
||||
@ -50,7 +53,7 @@
|
||||
"stream-mime-type": "^2.0.0",
|
||||
"turndown": "^7.2.0",
|
||||
"unstorage": "^1.15.0",
|
||||
"vite-plugin-static-copy": "^3.0.0",
|
||||
"vite-plugin-static-copy": "^3.1.2",
|
||||
"vue": "latest",
|
||||
"vue-router": "latest",
|
||||
"vue3-carousel": "^0.16.0",
|
||||
@ -86,5 +89,6 @@
|
||||
},
|
||||
"prisma": {
|
||||
"schema": "./prisma"
|
||||
}
|
||||
},
|
||||
"packageManager": "pnpm@10.15.0+sha512.486ebc259d3e999a4e8691ce03b5cac4a71cbeca39372a9b762cb500cfdf0873e2cb16abe3d951b1ee2cf012503f027b98b6584e4df22524e0c7450d9ec7aa7b"
|
||||
}
|
||||
|
||||
229
pages/account/tokens.vue
Normal file
229
pages/account/tokens.vue
Normal file
@ -0,0 +1,229 @@
|
||||
<template>
|
||||
<div>
|
||||
<div class="w-full flex justify-between items-center">
|
||||
<div>
|
||||
<h2
|
||||
class="mt-2 text-xl font-semibold tracking-tight text-zinc-100 sm:text-3xl"
|
||||
>
|
||||
{{ $t("account.token.title") }}
|
||||
</h2>
|
||||
<p
|
||||
class="mt-2 text-pretty text-sm font-medium text-zinc-400 sm:text-md/8"
|
||||
>
|
||||
{{ $t("account.token.subheader") }}
|
||||
</p>
|
||||
</div>
|
||||
<div>
|
||||
<LoadingButton :loading="false" @click="() => (createOpen = true)">
|
||||
{{ $t("common.create") }}
|
||||
</LoadingButton>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div
|
||||
v-if="newToken"
|
||||
class="mt-4 rounded-md p-4 bg-green-500/10 outline outline-green-500/20"
|
||||
>
|
||||
<div class="flex">
|
||||
<div class="shrink-0">
|
||||
<CheckCircleIcon class="size-5 text-green-400" aria-hidden="true" />
|
||||
</div>
|
||||
<div class="ml-3">
|
||||
<p class="text-sm font-medium text-green-300">
|
||||
{{ $t("account.token.success") }}
|
||||
</p>
|
||||
<p class="text-xs text-green-300/70">
|
||||
{{ $t("account.token.successNote") }}
|
||||
</p>
|
||||
<p
|
||||
class="mt-2 bg-zinc-950 px-3 py-2 font-mono text-zinc-400 rounded-xl"
|
||||
>
|
||||
{{ newToken }}
|
||||
</p>
|
||||
</div>
|
||||
<div class="ml-auto pl-3">
|
||||
<div class="-mx-1.5 -my-1.5">
|
||||
<button
|
||||
type="button"
|
||||
class="inline-flex rounded-md bg-green-50 p-1.5 text-green-500 hover:bg-green-100 focus-visible:ring-2 focus-visible:ring-green-600 focus-visible:ring-offset-2 focus-visible:ring-offset-green-50 focus-visible:outline-hidden dark:bg-transparent dark:text-green-400 dark:hover:bg-green-500/10 dark:focus-visible:ring-green-500 dark:focus-visible:ring-offset-1 dark:focus-visible:ring-offset-green-900"
|
||||
@click="() => (newToken = undefined)"
|
||||
>
|
||||
<span class="sr-only">{{ $t("common.close") }}</span>
|
||||
<XMarkIcon class="size-5" aria-hidden="true" />
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div
|
||||
class="mt-8 overflow-hidden rounded-xl border border-zinc-800 bg-zinc-900 shadow-sm"
|
||||
>
|
||||
<div class="overflow-x-auto">
|
||||
<table class="min-w-full divide-y divide-zinc-800">
|
||||
<thead>
|
||||
<tr class="bg-zinc-800/50">
|
||||
<th
|
||||
scope="col"
|
||||
class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-zinc-100 sm:pl-6"
|
||||
>
|
||||
{{ $t("common.name") }}
|
||||
</th>
|
||||
<th
|
||||
scope="col"
|
||||
class="px-3 py-3.5 text-left text-sm font-semibold text-zinc-100"
|
||||
>
|
||||
{{ $t("account.token.acls") }}
|
||||
</th>
|
||||
<th
|
||||
scope="col"
|
||||
class="px-3 py-3.5 text-left text-sm font-semibold text-zinc-100"
|
||||
>
|
||||
{{ $t("account.token.expiry") }}
|
||||
</th>
|
||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6">
|
||||
<span class="sr-only">{{ $t("actions") }}</span>
|
||||
</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody class="divide-y divide-zinc-800">
|
||||
<tr
|
||||
v-for="(token, tokenIdx) in tokens"
|
||||
:key="token.id"
|
||||
class="transition-colors duration-150 hover:bg-zinc-800/50"
|
||||
>
|
||||
<td
|
||||
class="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-zinc-100 sm:pl-6"
|
||||
>
|
||||
{{ token.name }}
|
||||
</td>
|
||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-zinc-400">
|
||||
<div class="flex flex-wrap gap-2">
|
||||
<span
|
||||
v-for="acl in token.acls"
|
||||
:key="acl"
|
||||
class="inline-flex items-center gap-x-1 rounded-md bg-blue-400/10 px-2 py-1 text-xs font-medium text-blue-400 ring-1 ring-inset ring-blue-400/20"
|
||||
>
|
||||
{{ acl }}
|
||||
</span>
|
||||
</div>
|
||||
</td>
|
||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-zinc-400">
|
||||
<RelativeTime v-if="token.expiresAt" :date="token.expiresAt" />
|
||||
<span v-else>{{ $t("account.token.noExpiry") }}</span>
|
||||
</td>
|
||||
<td
|
||||
class="relative whitespace-nowrap py-4 pl-3 pr-4 text-right text-sm font-medium sm:pr-6"
|
||||
>
|
||||
<button
|
||||
class="inline-flex items-center rounded-md bg-red-400/10 px-2 py-1 text-xs font-medium text-red-400 ring-1 ring-inset ring-red-400/20 transition-all duration-200 hover:bg-red-400/20 hover:scale-105 active:scale-95"
|
||||
@click="() => revokeToken(tokenIdx)"
|
||||
>
|
||||
{{ $t("account.token.revoke") }}
|
||||
<span class="sr-only">
|
||||
{{ $t("chars.srComma", [token.name]) }}
|
||||
</span>
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
<tr v-if="tokens.length === 0">
|
||||
<td colspan="5" class="py-8 text-center text-sm text-zinc-400">
|
||||
{{ $t("account.token.noTokens") }}
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<ModalCreateToken
|
||||
v-model="createOpen"
|
||||
:acls="acls"
|
||||
:loading="createLoading"
|
||||
:suggested-name="suggestedName"
|
||||
:suggested-acls="suggestedAcls"
|
||||
@create="(name, acls, expiry) => createToken(name, acls, expiry)"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { ArkErrors, type } from "arktype";
|
||||
import { DateTime, type DurationLike } from "luxon";
|
||||
import { CheckCircleIcon, XMarkIcon } from "@heroicons/vue/20/solid";
|
||||
|
||||
const tokens = ref(await $dropFetch("/api/v1/user/token"));
|
||||
const acls = await $dropFetch("/api/v1/user/token/acls");
|
||||
|
||||
const createOpen = ref(false);
|
||||
const createLoading = ref(false);
|
||||
|
||||
const newToken = ref<string | undefined>();
|
||||
|
||||
const suggestedName = ref();
|
||||
const suggestedAcls = ref<string[]>([]);
|
||||
|
||||
const payloadParser = type({
|
||||
name: "string?",
|
||||
acls: "string[]?",
|
||||
});
|
||||
|
||||
const route = useRoute();
|
||||
if (route.query.payload) {
|
||||
try {
|
||||
const rawPayload = JSON.parse(atob(route.query.payload.toString()));
|
||||
const payload = payloadParser(rawPayload);
|
||||
if (payload instanceof ArkErrors) throw payload;
|
||||
suggestedName.value = payload.name;
|
||||
suggestedAcls.value = payload.acls ?? [];
|
||||
createOpen.value = true;
|
||||
} catch {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "Failed to parse the token create payload.",
|
||||
fatal: true,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
async function createToken(
|
||||
name: string,
|
||||
acls: string[],
|
||||
expiry: DurationLike | undefined,
|
||||
) {
|
||||
createLoading.value = true;
|
||||
try {
|
||||
const result = await $dropFetch("/api/v1/user/token", {
|
||||
method: "POST",
|
||||
body: {
|
||||
name,
|
||||
acls,
|
||||
expiry: expiry ? DateTime.now().plus(expiry) : undefined,
|
||||
},
|
||||
failTitle: "Failed to create API token.",
|
||||
});
|
||||
console.log(result);
|
||||
newToken.value = result.token;
|
||||
tokens.value.push(result);
|
||||
} finally {
|
||||
/* empty */
|
||||
}
|
||||
createOpen.value = false;
|
||||
createLoading.value = false;
|
||||
}
|
||||
|
||||
async function revokeToken(index: number) {
|
||||
const token = tokens.value[index];
|
||||
if (!token) return;
|
||||
|
||||
await $dropFetch("/api/v1/user/token/:id", {
|
||||
method: "DELETE",
|
||||
params: {
|
||||
id: token.id,
|
||||
},
|
||||
failTitle: "Failed to revoke token.",
|
||||
});
|
||||
|
||||
tokens.value.splice(index, 1);
|
||||
}
|
||||
</script>
|
||||
@ -242,11 +242,40 @@
|
||||
{{ $t("common.noResults") }}
|
||||
</p>
|
||||
<p
|
||||
v-if="filteredLibraryGames.length == 0 && libraryGames.length == 0"
|
||||
v-if="
|
||||
filteredLibraryGames.length == 0 &&
|
||||
libraryGames.length == 0 &&
|
||||
libraryState.hasLibraries
|
||||
"
|
||||
class="text-zinc-600 text-sm font-display font-bold uppercase text-center col-span-4"
|
||||
>
|
||||
{{ $t("library.admin.noGames") }}
|
||||
</p>
|
||||
<p
|
||||
v-else-if="!libraryState.hasLibraries"
|
||||
class="flex flex-col gap-2 text-zinc-600 text-center col-span-4"
|
||||
>
|
||||
<span class="text-sm font-display font-bold uppercase">{{
|
||||
$t("library.admin.libraryHint")
|
||||
}}</span>
|
||||
|
||||
<NuxtLink
|
||||
class="transition text-xs text-zinc-600 hover:underline hover:text-zinc-400"
|
||||
href="https://docs.droposs.org/docs/library"
|
||||
target="_blank"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="library.admin.libraryHintDocsLink"
|
||||
tag="span"
|
||||
scope="global"
|
||||
class="inline-flex items-center gap-x-1"
|
||||
>
|
||||
<template #arrow>
|
||||
<ArrowTopRightOnSquareIcon class="size-4" />
|
||||
</template>
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
</p>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
@ -256,7 +285,11 @@ import {
|
||||
ExclamationTriangleIcon,
|
||||
ExclamationCircleIcon,
|
||||
} from "@heroicons/vue/16/solid";
|
||||
import { InformationCircleIcon, StarIcon } from "@heroicons/vue/20/solid";
|
||||
import {
|
||||
ArrowTopRightOnSquareIcon,
|
||||
InformationCircleIcon,
|
||||
StarIcon,
|
||||
} from "@heroicons/vue/20/solid";
|
||||
import { MagnifyingGlassIcon } from "@heroicons/vue/24/outline";
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
@ -64,8 +64,14 @@
|
||||
>
|
||||
{{ source.name }}
|
||||
</td>
|
||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-zinc-400">
|
||||
{{ source.backend }}
|
||||
<td
|
||||
class="whitespace-nowrap px-3 py-4 text-sm text-zinc-400 inline-flex gap-x-1 items-center"
|
||||
>
|
||||
<component
|
||||
:is="optionsMetadata[source.backend].icon"
|
||||
class="size-5 text-zinc-400"
|
||||
/>
|
||||
{{ optionsMetadata[source.backend].title }}
|
||||
</td>
|
||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-zinc-400">
|
||||
<CheckIcon
|
||||
@ -189,11 +195,34 @@
|
||||
<RadioGroupLabel
|
||||
as="span"
|
||||
class="font-semibold text-zinc-100"
|
||||
>{{ source }}</RadioGroupLabel
|
||||
>{{ metadata.title }}
|
||||
<span class="ml-2 font-mono text-zinc-500 text-xs">{{
|
||||
source
|
||||
}}</span></RadioGroupLabel
|
||||
>
|
||||
<RadioGroupDescription
|
||||
as="span"
|
||||
class="text-zinc-400 text-xs"
|
||||
>
|
||||
<RadioGroupDescription as="span" class="text-zinc-400">
|
||||
{{ metadata.description }}
|
||||
</RadioGroupDescription>
|
||||
<NuxtLink
|
||||
:href="metadata.docsLink"
|
||||
:external="true"
|
||||
target="_blank"
|
||||
class="mt-2 block w-fit rounded-md bg-blue-600 px-2 py-1 text-center text-xs font-semibold text-white shadow-sm hover:bg-blue-500 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-blue-600"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="library.admin.sources.documentationLink"
|
||||
tag="span"
|
||||
scope="global"
|
||||
class="inline-flex items-center gap-x-1"
|
||||
>
|
||||
<template #arrow>
|
||||
<ArrowTopRightOnSquareIcon class="size-4" />
|
||||
</template>
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
</span>
|
||||
</span>
|
||||
<span
|
||||
@ -269,6 +298,7 @@
|
||||
*/
|
||||
|
||||
import {
|
||||
DropLogo,
|
||||
SourceOptionsFilesystem,
|
||||
SourceOptionsFlatFilesystem,
|
||||
} from "#components";
|
||||
@ -279,8 +309,11 @@ import {
|
||||
RadioGroupLabel,
|
||||
RadioGroupOption,
|
||||
} from "@headlessui/vue";
|
||||
import { XCircleIcon } from "@heroicons/vue/20/solid";
|
||||
import { CheckIcon, DocumentIcon, XMarkIcon } from "@heroicons/vue/24/outline";
|
||||
import {
|
||||
XCircleIcon,
|
||||
ArrowTopRightOnSquareIcon,
|
||||
} from "@heroicons/vue/20/solid";
|
||||
import { BackwardIcon, CheckIcon, XMarkIcon } from "@heroicons/vue/24/outline";
|
||||
import { FetchError } from "ofetch";
|
||||
import type { Component } from "vue";
|
||||
import type { LibraryBackend } from "~/prisma/client/enums";
|
||||
@ -324,17 +357,23 @@ const optionUIs: { [key in LibraryBackend]: Component } = {
|
||||
};
|
||||
const optionsMetadata: {
|
||||
[key in LibraryBackend]: {
|
||||
title: string;
|
||||
description: string;
|
||||
docsLink: string;
|
||||
icon: Component;
|
||||
};
|
||||
} = {
|
||||
Filesystem: {
|
||||
title: t("library.admin.sources.fsTitle"),
|
||||
description: t("library.admin.sources.fsDesc"),
|
||||
icon: DocumentIcon,
|
||||
docsLink: "https://docs.droposs.org/docs/library#drop-style",
|
||||
icon: DropLogo,
|
||||
},
|
||||
FlatFilesystem: {
|
||||
title: t("library.admin.sources.fsFlatTitle"),
|
||||
description: t("library.admin.sources.fsFlatDesc"),
|
||||
icon: DocumentIcon,
|
||||
docsLink: "https://docs.droposs.org/docs/library#flat-style-or-compat",
|
||||
icon: BackwardIcon,
|
||||
},
|
||||
};
|
||||
const optionsMetadataIter = Object.entries(optionsMetadata);
|
||||
|
||||
@ -30,7 +30,7 @@
|
||||
{{ company.mName }}
|
||||
<button @click="() => editName()">
|
||||
<PencilIcon
|
||||
class="transition duration-200 opacity-0 group-hover/name:opacity-100 size-8"
|
||||
class="transition duration-200 xl:opacity-0 group-hover/name:opacity-100 size-8"
|
||||
/>
|
||||
</button>
|
||||
</h1>
|
||||
@ -43,17 +43,20 @@
|
||||
}}
|
||||
<button @click="() => editShortDescription()">
|
||||
<PencilIcon
|
||||
class="transition duration-200 opacity-0 group-hover/description:opacity-100 size-5"
|
||||
class="transition duration-200 xl:opacity-0 group-hover/description:opacity-100 size-5"
|
||||
/>
|
||||
</button>
|
||||
</p>
|
||||
<p
|
||||
class="group/website mt-1 text-zinc-500 inline-flex items-center gap-x-3"
|
||||
>
|
||||
{{ company.mWebsite }}
|
||||
{{
|
||||
company.mWebsite ||
|
||||
$t("library.admin.metadata.companies.editor.websitePlaceholder")
|
||||
}}
|
||||
<button @click="() => editWebsite()">
|
||||
<PencilIcon
|
||||
class="transition duration-200 opacity-0 group-hover/website:opacity-100 size-4"
|
||||
class="transition duration-200 xl:opacity-0 group-hover/website:opacity-100 size-4"
|
||||
/>
|
||||
</button>
|
||||
</p>
|
||||
|
||||
@ -10,20 +10,12 @@
|
||||
</p>
|
||||
</div>
|
||||
<div class="mt-4 sm:ml-16 sm:mt-0 sm:flex-none">
|
||||
<NuxtLink
|
||||
to="/admin/library/sources"
|
||||
<button
|
||||
class="block rounded-md bg-blue-600 px-3 py-2 text-center text-sm font-semibold text-white shadow-sm transition-all duration-200 hover:bg-blue-500 hover:scale-105 hover:shadow-lg hover:shadow-blue-500/25 active:scale-95 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-blue-600"
|
||||
@click="() => (createCompanyOpen = true)"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="library.admin.sources.link"
|
||||
tag="span"
|
||||
scope="global"
|
||||
>
|
||||
<template #arrow>
|
||||
<span aria-hidden="true">{{ $t("chars.arrow") }}</span>
|
||||
</template>
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
{{ $t("common.create") }}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="mt-2 grid grid-cols-1">
|
||||
@ -105,6 +97,10 @@
|
||||
{{ $t("library.admin.metadata.companies.noCompanies") }}
|
||||
</p>
|
||||
</ul>
|
||||
<ModalCreateCompany
|
||||
v-model="createCompanyOpen"
|
||||
@created="(company) => createCompany(company)"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -122,9 +118,12 @@ useHead({
|
||||
title: t("library.admin.metadata.companies.title"),
|
||||
});
|
||||
|
||||
const createCompanyOpen = ref(false);
|
||||
|
||||
const searchQuery = ref("");
|
||||
|
||||
const companies = ref(await $dropFetch("/api/v1/admin/company"));
|
||||
const rawCompanies = await $dropFetch("/api/v1/admin/company");
|
||||
const companies = ref(rawCompanies);
|
||||
|
||||
const filteredCompanies = computed(() =>
|
||||
companies.value.filter((e: CompanyModel) => {
|
||||
@ -147,4 +146,8 @@ async function deleteCompany(id: string) {
|
||||
const index = companies.value.findIndex((e) => e.id === id);
|
||||
companies.value.splice(index, 1);
|
||||
}
|
||||
|
||||
function createCompany(company: (typeof companies.value)[number]) {
|
||||
companies.value.push(company);
|
||||
}
|
||||
</script>
|
||||
|
||||
68
pages/admin/settings.vue
Normal file
68
pages/admin/settings.vue
Normal file
@ -0,0 +1,68 @@
|
||||
<template>
|
||||
<div class="flex flex-col">
|
||||
<!-- tabs-->
|
||||
<div>
|
||||
<div class="border-b border-gray-200 dark:border-white/10">
|
||||
<nav class="-mb-px flex gap-x-2" aria-label="Tabs">
|
||||
<NuxtLink
|
||||
v-for="(tab, tabIdx) in navigation"
|
||||
:key="tab.route"
|
||||
:href="tab.route"
|
||||
:class="[
|
||||
currentNavigationIndex == tabIdx
|
||||
? 'border-blue-500 text-blue-600 dark:border-blue-400 dark:text-blue-400'
|
||||
: 'border-transparent text-gray-500 hover:border-gray-300 hover:text-gray-700 dark:text-gray-400 dark:hover:border-white/20 dark:hover:text-gray-300',
|
||||
'group inline-flex items-center border-b-2 px-1 py-4 text-sm font-medium',
|
||||
]"
|
||||
:aria-current="tab ? 'page' : undefined"
|
||||
>
|
||||
<component
|
||||
:is="tab.icon"
|
||||
:class="[
|
||||
currentNavigationIndex == tabIdx
|
||||
? 'text-blue-500 dark:text-blue-400'
|
||||
: 'text-gray-400 group-hover:text-gray-500 dark:text-gray-500 dark:group-hover:text-gray-400',
|
||||
'mr-2 -ml-0.5 size-5',
|
||||
]"
|
||||
aria-hidden="true"
|
||||
/>
|
||||
<span>{{ tab.label }}</span>
|
||||
</NuxtLink>
|
||||
</nav>
|
||||
</div>
|
||||
</div>
|
||||
<!-- content -->
|
||||
<div class="mt-4 grow flex">
|
||||
<NuxtPage />
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import {
|
||||
BuildingStorefrontIcon,
|
||||
CodeBracketIcon,
|
||||
} from "@heroicons/vue/24/outline";
|
||||
|
||||
const navigation: Array<NavigationItem & { icon: Component }> = [
|
||||
{
|
||||
label: $t("header.admin.settings.store"),
|
||||
route: "/admin/settings",
|
||||
prefix: "/admin/settings",
|
||||
icon: BuildingStorefrontIcon,
|
||||
},
|
||||
{
|
||||
label: $t("header.admin.settings.tokens"),
|
||||
route: "/admin/settings/tokens",
|
||||
prefix: "/admin/settings/tokens",
|
||||
icon: CodeBracketIcon,
|
||||
},
|
||||
];
|
||||
|
||||
// const notifications = useNotifications();
|
||||
// const unreadNotifications = computed(() =>
|
||||
// notifications.value.filter((e) => !e.read)
|
||||
// );
|
||||
|
||||
const currentNavigationIndex = useCurrentNavigationIndex(navigation);
|
||||
</script>
|
||||
@ -1,68 +1,55 @@
|
||||
<template>
|
||||
<div class="space-y-4">
|
||||
<div class="sm:flex sm:items-center">
|
||||
<div class="sm:flex-auto">
|
||||
<h1 class="text-2xl font-semibold text-zinc-100">
|
||||
{{ $t("settings.admin.title") }}
|
||||
</h1>
|
||||
<p class="mt-2 text-base text-zinc-400">
|
||||
{{ $t("settings.admin.description") }}
|
||||
</p>
|
||||
</div>
|
||||
<form class="space-y-4" @submit.prevent="() => saveSettings()">
|
||||
<div class="pb-4 border-b border-zinc-700">
|
||||
<h2 class="text-xl font-semibold text-zinc-100">
|
||||
{{ $t("settings.admin.store.title") }}
|
||||
</h2>
|
||||
|
||||
<h3 class="text-base font-medium text-zinc-400 mb-3 m-x-0">
|
||||
{{ $t("settings.admin.store.showGamePanelTextDecoration") }}
|
||||
</h3>
|
||||
<ul class="flex gap-3">
|
||||
<li class="inline-block">
|
||||
<OptionWrapper
|
||||
:active="showGamePanelTextDecoration"
|
||||
@click="setShowTitleDescription(true)"
|
||||
>
|
||||
<div class="flex">
|
||||
<GamePanel
|
||||
:animate="false"
|
||||
:game="game"
|
||||
:default-placeholder="true"
|
||||
/>
|
||||
</div>
|
||||
</OptionWrapper>
|
||||
</li>
|
||||
<li class="inline-block">
|
||||
<OptionWrapper
|
||||
:active="!showGamePanelTextDecoration"
|
||||
@click="setShowTitleDescription(false)"
|
||||
>
|
||||
<div class="flex">
|
||||
<GamePanel
|
||||
:game="game"
|
||||
:show-title-description="false"
|
||||
:animate="false"
|
||||
:default-placeholder="true"
|
||||
/>
|
||||
</div>
|
||||
</OptionWrapper>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<form class="space-y-4" @submit.prevent="() => saveSettings()">
|
||||
<div class="py-6 border-y border-zinc-700">
|
||||
<h2 class="text-xl font-semibold text-zinc-100">
|
||||
{{ $t("settings.admin.store.title") }}
|
||||
</h2>
|
||||
|
||||
<h3 class="text-base font-medium text-zinc-400 mb-3 m-x-0">
|
||||
{{ $t("settings.admin.store.showGamePanelTextDecoration") }}
|
||||
</h3>
|
||||
<ul class="flex gap-3">
|
||||
<li class="inline-block">
|
||||
<OptionWrapper
|
||||
:active="showGamePanelTextDecoration"
|
||||
@click="setShowTitleDescription(true)"
|
||||
>
|
||||
<div class="flex">
|
||||
<GamePanel
|
||||
:animate="false"
|
||||
:game="game"
|
||||
:default-placeholder="true"
|
||||
/>
|
||||
</div>
|
||||
</OptionWrapper>
|
||||
</li>
|
||||
<li class="inline-block">
|
||||
<OptionWrapper
|
||||
:active="!showGamePanelTextDecoration"
|
||||
@click="setShowTitleDescription(false)"
|
||||
>
|
||||
<div class="flex">
|
||||
<GamePanel
|
||||
:game="game"
|
||||
:show-title-description="false"
|
||||
:animate="false"
|
||||
:default-placeholder="true"
|
||||
/>
|
||||
</div>
|
||||
</OptionWrapper>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<LoadingButton
|
||||
type="submit"
|
||||
class="inline-flex w-full shadow-sm sm:w-auto"
|
||||
:loading="saving"
|
||||
:disabled="!allowSave"
|
||||
>
|
||||
{{ allowSave ? $t("common.save") : $t("common.saved") }}
|
||||
</LoadingButton>
|
||||
</form>
|
||||
</div>
|
||||
<LoadingButton
|
||||
type="submit"
|
||||
class="inline-flex w-full shadow-sm sm:w-auto"
|
||||
:loading="saving"
|
||||
:disabled="!allowSave"
|
||||
>
|
||||
{{ allowSave ? $t("common.save") : $t("common.saved") }}
|
||||
</LoadingButton>
|
||||
</form>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
|
||||
233
pages/admin/settings/tokens.vue
Normal file
233
pages/admin/settings/tokens.vue
Normal file
@ -0,0 +1,233 @@
|
||||
<template>
|
||||
<div class="w-full">
|
||||
<div class="w-full flex justify-between items-center">
|
||||
<div>
|
||||
<h2
|
||||
class="mt-2 text-xl font-semibold tracking-tight text-zinc-100 sm:text-3xl"
|
||||
>
|
||||
{{ $t("account.token.title") }}
|
||||
</h2>
|
||||
<p
|
||||
class="mt-2 text-pretty text-sm font-medium text-zinc-400 sm:text-md/8"
|
||||
>
|
||||
{{ $t("account.token.subheader") }}
|
||||
</p>
|
||||
</div>
|
||||
<div>
|
||||
<LoadingButton :loading="false" @click="() => (createOpen = true)">
|
||||
{{ $t("common.create") }}
|
||||
</LoadingButton>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div
|
||||
v-if="newToken"
|
||||
class="mt-4 rounded-md p-4 bg-green-500/10 outline outline-green-500/20"
|
||||
>
|
||||
<div class="flex">
|
||||
<div class="shrink-0">
|
||||
<CheckCircleIcon class="size-5 text-green-400" aria-hidden="true" />
|
||||
</div>
|
||||
<div class="ml-3">
|
||||
<p class="text-sm font-medium text-green-300">
|
||||
{{ $t("account.token.success") }}
|
||||
</p>
|
||||
<p class="text-xs text-green-300/70">
|
||||
{{ $t("account.token.successNote") }}
|
||||
</p>
|
||||
<p
|
||||
class="mt-2 bg-zinc-950 px-3 py-2 font-mono text-zinc-400 rounded-xl"
|
||||
>
|
||||
{{ newToken }}
|
||||
</p>
|
||||
</div>
|
||||
<div class="ml-auto pl-3">
|
||||
<div class="-mx-1.5 -my-1.5">
|
||||
<button
|
||||
type="button"
|
||||
class="inline-flex rounded-md bg-green-50 p-1.5 text-green-500 hover:bg-green-100 focus-visible:ring-2 focus-visible:ring-green-600 focus-visible:ring-offset-2 focus-visible:ring-offset-green-50 focus-visible:outline-hidden dark:bg-transparent dark:text-green-400 dark:hover:bg-green-500/10 dark:focus-visible:ring-green-500 dark:focus-visible:ring-offset-1 dark:focus-visible:ring-offset-green-900"
|
||||
@click="() => (newToken = undefined)"
|
||||
>
|
||||
<span class="sr-only">{{ $t("common.close") }}</span>
|
||||
<XMarkIcon class="size-5" aria-hidden="true" />
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div
|
||||
class="mt-8 overflow-hidden rounded-xl border border-zinc-800 bg-zinc-900 shadow-sm"
|
||||
>
|
||||
<div class="overflow-x-auto">
|
||||
<table class="min-w-full divide-y divide-zinc-800">
|
||||
<thead>
|
||||
<tr class="bg-zinc-800/50">
|
||||
<th
|
||||
scope="col"
|
||||
class="py-3.5 pl-4 pr-3 text-left text-sm font-semibold text-zinc-100 sm:pl-6"
|
||||
>
|
||||
{{ $t("common.name") }}
|
||||
</th>
|
||||
<th
|
||||
scope="col"
|
||||
class="px-3 py-3.5 text-left text-sm font-semibold text-zinc-100"
|
||||
>
|
||||
{{ $t("account.token.acls") }}
|
||||
</th>
|
||||
<th
|
||||
scope="col"
|
||||
class="px-3 py-3.5 text-left text-sm font-semibold text-zinc-100"
|
||||
>
|
||||
{{ $t("account.token.expiry") }}
|
||||
</th>
|
||||
<th scope="col" class="relative py-3.5 pl-3 pr-4 sm:pr-6">
|
||||
<span class="sr-only">{{ $t("actions") }}</span>
|
||||
</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody class="divide-y divide-zinc-800">
|
||||
<tr
|
||||
v-for="(token, tokenIdx) in tokens"
|
||||
:key="token.id"
|
||||
class="transition-colors duration-150 hover:bg-zinc-800/50"
|
||||
>
|
||||
<td
|
||||
class="whitespace-nowrap py-4 pl-4 pr-3 text-sm font-medium text-zinc-100 sm:pl-6"
|
||||
>
|
||||
{{ token.name }}
|
||||
</td>
|
||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-zinc-400">
|
||||
<div class="flex flex-wrap gap-2">
|
||||
<span
|
||||
v-for="acl in token.acls"
|
||||
:key="acl"
|
||||
class="inline-flex items-center gap-x-1 rounded-md bg-blue-400/10 px-2 py-1 text-xs font-medium text-blue-400 ring-1 ring-inset ring-blue-400/20"
|
||||
>
|
||||
{{ acl }}
|
||||
</span>
|
||||
</div>
|
||||
</td>
|
||||
<td class="whitespace-nowrap px-3 py-4 text-sm text-zinc-400">
|
||||
<RelativeTime v-if="token.expiresAt" :date="token.expiresAt" />
|
||||
<span v-else>{{ $t("account.token.noExpiry") }}</span>
|
||||
</td>
|
||||
<td
|
||||
class="relative whitespace-nowrap py-4 pl-3 pr-4 text-right text-sm font-medium sm:pr-6"
|
||||
>
|
||||
<button
|
||||
class="inline-flex items-center rounded-md bg-red-400/10 px-2 py-1 text-xs font-medium text-red-400 ring-1 ring-inset ring-red-400/20 transition-all duration-200 hover:bg-red-400/20 hover:scale-105 active:scale-95"
|
||||
@click="() => revokeToken(tokenIdx)"
|
||||
>
|
||||
{{ $t("account.token.revoke") }}
|
||||
<span class="sr-only">
|
||||
{{ $t("chars.srComma", [token.name]) }}
|
||||
</span>
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
<tr v-if="tokens.length === 0">
|
||||
<td colspan="5" class="py-8 text-center text-sm text-zinc-400">
|
||||
{{ $t("account.token.noTokens") }}
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<ModalCreateToken
|
||||
v-model="createOpen"
|
||||
:acls="acls"
|
||||
:loading="createLoading"
|
||||
:suggested-name="suggestedName"
|
||||
:suggested-acls="suggestedAcls"
|
||||
@create="(name, acls, expiry) => createToken(name, acls, expiry)"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { ArkErrors, type } from "arktype";
|
||||
import { DateTime, type DurationLike } from "luxon";
|
||||
import { CheckCircleIcon, XMarkIcon } from "@heroicons/vue/20/solid";
|
||||
|
||||
definePageMeta({
|
||||
layout: "admin",
|
||||
});
|
||||
|
||||
const tokens = ref(await $dropFetch("/api/v1/admin/token"));
|
||||
const acls = await $dropFetch("/api/v1/admin/token/acls");
|
||||
|
||||
const createOpen = ref(false);
|
||||
const createLoading = ref(false);
|
||||
|
||||
const newToken = ref<string | undefined>();
|
||||
|
||||
const suggestedName = ref();
|
||||
const suggestedAcls = ref<string[]>([]);
|
||||
|
||||
const payloadParser = type({
|
||||
name: "string?",
|
||||
acls: "string[]?",
|
||||
});
|
||||
|
||||
const route = useRoute();
|
||||
if (route.query.payload) {
|
||||
try {
|
||||
const rawPayload = JSON.parse(atob(route.query.payload.toString()));
|
||||
const payload = payloadParser(rawPayload);
|
||||
if (payload instanceof ArkErrors) throw payload;
|
||||
suggestedName.value = payload.name;
|
||||
suggestedAcls.value = payload.acls ?? [];
|
||||
createOpen.value = true;
|
||||
} catch {
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "Failed to parse the token create payload.",
|
||||
fatal: true,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
async function createToken(
|
||||
name: string,
|
||||
acls: string[],
|
||||
expiry: DurationLike | undefined,
|
||||
) {
|
||||
createLoading.value = true;
|
||||
try {
|
||||
const result = await $dropFetch("/api/v1/admin/token", {
|
||||
method: "POST",
|
||||
body: {
|
||||
name,
|
||||
acls,
|
||||
expiry: expiry ? DateTime.now().plus(expiry) : undefined,
|
||||
},
|
||||
failTitle: "Failed to create API token.",
|
||||
});
|
||||
console.log(result);
|
||||
newToken.value = result.token;
|
||||
tokens.value.push(result);
|
||||
} catch {
|
||||
/* empty */
|
||||
}
|
||||
createOpen.value = false;
|
||||
createLoading.value = false;
|
||||
}
|
||||
|
||||
async function revokeToken(index: number) {
|
||||
const token = tokens.value[index];
|
||||
if (!token) return;
|
||||
|
||||
await $dropFetch("/api/v1/admin/token/:id", {
|
||||
method: "DELETE",
|
||||
params: {
|
||||
id: token.id,
|
||||
},
|
||||
failTitle: "Failed to revoke token.",
|
||||
});
|
||||
|
||||
tokens.value.splice(index, 1);
|
||||
}
|
||||
</script>
|
||||
@ -44,19 +44,26 @@
|
||||
</div>
|
||||
{{ task.name }}
|
||||
</h1>
|
||||
<div class="h-2 rounded-full bg-zinc-950 overflow-hidden">
|
||||
<div
|
||||
class="bg-zinc-950 p-2 rounded-md h-[80vh] flex flex-col flex-col-reverse overflow-y-scroll gap-y-1"
|
||||
>
|
||||
<LogLine
|
||||
v-for="(_, idx) in task.log"
|
||||
:key="idx"
|
||||
:log="parseTaskLog(task.log.at(-(idx + 1)))"
|
||||
/>
|
||||
</div>
|
||||
<div class="relative h-5 rounded-xl bg-zinc-950 overflow-hidden">
|
||||
<div
|
||||
:style="{ width: `${task.progress}%` }"
|
||||
class="transition-all bg-blue-600 h-full"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div
|
||||
class="relative bg-zinc-950/50 rounded-md p-2 text-zinc-100 h-[80vh] overflow-y-scroll"
|
||||
>
|
||||
<pre v-for="(line, idx) in task.log" :key="idx">{{
|
||||
formatLine(line)
|
||||
}}</pre>
|
||||
<span
|
||||
class="absolute inset-0 flex items-center justify-center text-blue-200 text-sm font-bold font-display"
|
||||
>{{
|
||||
$t("tasks.admin.progress", [Math.round(task.progress * 10) / 10])
|
||||
}}</span
|
||||
>
|
||||
</div>
|
||||
</div>
|
||||
<div v-else role="status" class="w-full flex items-center justify-center">
|
||||
@ -90,11 +97,6 @@ const taskId = route.params.id.toString();
|
||||
|
||||
const task = useTask(taskId);
|
||||
|
||||
function formatLine(line: string): string {
|
||||
const res = parseTaskLog(line);
|
||||
return `[${res.timestamp}] ${res.message}`;
|
||||
}
|
||||
|
||||
definePageMeta({
|
||||
layout: "admin",
|
||||
});
|
||||
|
||||
@ -13,62 +13,7 @@
|
||||
:key="task.value?.id"
|
||||
class="col-span-1 divide-y divide-gray-200 rounded-lg bg-zinc-800 border border-zinc-700 shadow-sm"
|
||||
>
|
||||
<div
|
||||
v-if="task.value"
|
||||
class="flex w-full items-center justify-between space-x-6 p-6"
|
||||
>
|
||||
<div class="flex-1 truncate">
|
||||
<div class="flex items-center space-x-2">
|
||||
<div>
|
||||
<CheckIcon
|
||||
v-if="task.value.success"
|
||||
class="size-4 text-green-600"
|
||||
/>
|
||||
<XMarkIcon
|
||||
v-else-if="task.value.error"
|
||||
class="size-4 text-red-600"
|
||||
/>
|
||||
<div
|
||||
v-else
|
||||
class="size-2 bg-blue-600 rounded-full animate-pulse"
|
||||
/>
|
||||
</div>
|
||||
<h3 class="truncate text-sm font-medium text-zinc-100">
|
||||
{{ task.value.name }}
|
||||
</h3>
|
||||
</div>
|
||||
<p class="text-xs text-zinc-600 mt-0.5 font-mono">
|
||||
{{ task.value.id }}
|
||||
</p>
|
||||
<div class="mt-1 w-full rounded-full overflow-hidden bg-zinc-900">
|
||||
<div
|
||||
:style="{ width: `${task.value.progress}%` }"
|
||||
class="bg-blue-600 h-1.5 transition-all"
|
||||
/>
|
||||
</div>
|
||||
<p class="mt-1 truncate text-sm text-zinc-400">
|
||||
{{ parseTaskLog(task.value.log.at(-1) ?? "").message }}
|
||||
</p>
|
||||
<NuxtLink
|
||||
type="button"
|
||||
:href="`/admin/task/${task.value.id}`"
|
||||
class="mt-3 rounded-md text-xs font-medium text-zinc-100 hover:text-zinc-300 focus:outline-none focus:ring-2 focus:ring-zinc-100 focus:ring-offset-2"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="tasks.admin.viewTask"
|
||||
tag="span"
|
||||
scope="global"
|
||||
>
|
||||
<template #arrow>
|
||||
<span aria-hidden="true">{{ $t("chars.arrow") }}</span>
|
||||
</template>
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
</div>
|
||||
</div>
|
||||
<div v-else>
|
||||
<!-- renders server side when we don't want to access the current tasks -->
|
||||
</div>
|
||||
<TaskWidget :task="task.value" :active="true" />
|
||||
</li>
|
||||
</ul>
|
||||
<div
|
||||
@ -89,51 +34,7 @@
|
||||
:key="task.id"
|
||||
class="col-span-1 divide-y divide-gray-200 rounded-lg bg-zinc-800 border border-zinc-700 shadow-sm"
|
||||
>
|
||||
<div class="flex w-full items-center justify-between space-x-6 p-6">
|
||||
<div class="flex-1 truncate">
|
||||
<div class="flex items-center space-x-2">
|
||||
<div>
|
||||
<CheckIcon
|
||||
v-if="task.success"
|
||||
class="size-4 text-green-600"
|
||||
/>
|
||||
<XMarkIcon
|
||||
v-else-if="task.error"
|
||||
class="size-4 text-red-600"
|
||||
/>
|
||||
<div
|
||||
v-else
|
||||
class="size-2 bg-blue-600 rounded-full animate-pulse"
|
||||
/>
|
||||
</div>
|
||||
<h3 class="truncate text-sm font-medium text-zinc-100">
|
||||
{{ task.name }}
|
||||
</h3>
|
||||
<RelativeTime class="text-zinc-500" :date="task.ended" />
|
||||
</div>
|
||||
<p class="text-xs text-zinc-600 mt-0.5 font-mono">
|
||||
{{ task.id }}
|
||||
</p>
|
||||
<p class="mt-1 truncate text-sm text-zinc-400">
|
||||
{{ parseTaskLog(task.log.at(-1) ?? "").message }}
|
||||
</p>
|
||||
<NuxtLink
|
||||
type="button"
|
||||
:href="`/admin/task/${task.id}`"
|
||||
class="mt-3 rounded-md text-xs font-medium text-zinc-100 hover:text-zinc-300 focus:outline-none focus:ring-2 focus:ring-zinc-100 focus:ring-offset-2"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="tasks.admin.viewTask"
|
||||
tag="span"
|
||||
scope="global"
|
||||
>
|
||||
<template #arrow>
|
||||
<span aria-hidden="true">{{ $t("chars.arrow") }}</span>
|
||||
</template>
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
</div>
|
||||
</div>
|
||||
<TaskWidget :task="task" />
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
@ -157,6 +58,21 @@
|
||||
<p class="mt-1 text-sm text-zinc-400">
|
||||
{{ scheduledTasks[task].description }}
|
||||
</p>
|
||||
<button
|
||||
class="mt-3 rounded-md text-xs font-medium text-zinc-100 hover:text-zinc-300 focus:outline-none focus:ring-2 focus:ring-zinc-100 focus:ring-offset-2"
|
||||
@click="() => startTask(task)"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="tasks.admin.execute"
|
||||
tag="span"
|
||||
scope="global"
|
||||
class="inline-flex items-center gap-x-1"
|
||||
>
|
||||
<template #arrow>
|
||||
<PlayIcon class="size-4" aria-hidden="true" />
|
||||
</template>
|
||||
</i18n-t>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</li>
|
||||
@ -180,6 +96,21 @@
|
||||
<p class="mt-1 text-sm text-zinc-400">
|
||||
{{ scheduledTasks[task].description }}
|
||||
</p>
|
||||
<button
|
||||
class="mt-3 rounded-md text-xs font-medium text-zinc-100 hover:text-zinc-300 focus:outline-none focus:ring-2 focus:ring-zinc-100 focus:ring-offset-2"
|
||||
@click="() => startTask(task)"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="tasks.admin.execute"
|
||||
tag="span"
|
||||
scope="global"
|
||||
class="inline-flex items-center gap-x-1"
|
||||
>
|
||||
<template #arrow>
|
||||
<PlayIcon class="size-4" aria-hidden="true" />
|
||||
</template>
|
||||
</i18n-t>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</li>
|
||||
@ -189,7 +120,7 @@
|
||||
</div>
|
||||
</template>
|
||||
<script lang="ts" setup>
|
||||
import { CheckIcon, XMarkIcon } from "@heroicons/vue/24/outline";
|
||||
import { PlayIcon } from "@heroicons/vue/24/outline";
|
||||
import type { TaskGroup } from "~/server/internal/tasks/group";
|
||||
|
||||
useHead({
|
||||
@ -205,7 +136,9 @@ const { t } = useI18n();
|
||||
const { runningTasks, historicalTasks, dailyTasks, weeklyTasks } =
|
||||
await $dropFetch("/api/v1/admin/task");
|
||||
|
||||
const liveRunningTasks = await Promise.all(runningTasks.map((e) => useTask(e)));
|
||||
const liveRunningTasks = ref(
|
||||
await Promise.all(runningTasks.map((e) => useTask(e))),
|
||||
);
|
||||
|
||||
const scheduledTasks: {
|
||||
[key in TaskGroup]: { name: string; description: string };
|
||||
@ -230,5 +163,19 @@ const scheduledTasks: {
|
||||
name: "",
|
||||
description: "",
|
||||
},
|
||||
debug: {
|
||||
name: "Debug Task",
|
||||
description: "Does debugging things.",
|
||||
},
|
||||
};
|
||||
|
||||
async function startTask(taskGroup: string) {
|
||||
const task = await $dropFetch("/api/v1/admin/task", {
|
||||
method: "POST",
|
||||
body: { taskGroup },
|
||||
failTitle: "Failed to start task",
|
||||
});
|
||||
const taskRef = await useTask(task.id);
|
||||
liveRunningTasks.value.push(taskRef);
|
||||
}
|
||||
</script>
|
||||
|
||||
@ -72,7 +72,7 @@
|
||||
{{ $t("store.images") }}
|
||||
</h2>
|
||||
<div class="relative">
|
||||
<VueCarousel :items-to-show="1">
|
||||
<VueCarousel :items-to-show="1" :wrap-around="true">
|
||||
<VueSlide
|
||||
v-for="image in game.mImageCarouselObjectIds"
|
||||
:key="image"
|
||||
|
||||
@ -85,9 +85,15 @@
|
||||
v-else-if="!useModal"
|
||||
class="bg-zinc-950/30 flex items-center justify-center"
|
||||
>
|
||||
<p class="uppercase text-sm font-display text-zinc-700 font-bold">
|
||||
<!-- <p class="uppercase text-sm font-display text-zinc-700 font-bold">
|
||||
{{ $t("setup.noPage") }}
|
||||
</p>
|
||||
</p> -->
|
||||
<img
|
||||
src="/wallpapers/signin.jpg"
|
||||
class="inset-0 h-full w-full object-cover"
|
||||
alt=""
|
||||
preload
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
|
||||
@ -183,7 +183,7 @@
|
||||
{{ game.mShortDescription }}
|
||||
</p>
|
||||
<div class="mt-6 py-4 rounded">
|
||||
<VueCarousel :items-to-show="1">
|
||||
<VueCarousel :items-to-show="1" :wrap-around="true">
|
||||
<VueSlide
|
||||
v-for="image in game.mImageCarouselObjectIds"
|
||||
:key="image"
|
||||
|
||||
@ -59,13 +59,30 @@
|
||||
</VueCarousel>
|
||||
<div
|
||||
v-else
|
||||
class="w-full h-full flex items-center justify-center bg-zinc-950/50 px-6 py-32 sm:px-12 sm:py-40 lg:px-16"
|
||||
class="w-full h-full flex flex-col items-center justify-center bg-zinc-950/50 px-6 py-32 sm:px-12 sm:py-40 lg:px-16 gap-4"
|
||||
>
|
||||
<h2
|
||||
class="uppercase text-xl font-bold tracking-tight text-zinc-700 sm:text-3xl"
|
||||
>
|
||||
{{ $t("store.noGame") }}
|
||||
{{ $t("store.noFeatured") }}
|
||||
</h2>
|
||||
<NuxtLink
|
||||
v-if="user?.admin"
|
||||
to="/admin/library"
|
||||
type="button"
|
||||
class="inline-flex items-center gap-x-2 rounded-md bg-zinc-800 px-3 py-1 text-sm font-semibold font-display text-white shadow-sm hover:bg-zinc-700 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-blue-600 duration-200 hover:scale-105 active:scale-95"
|
||||
>
|
||||
<i18n-t
|
||||
keypath="store.openFeatured"
|
||||
tag="span"
|
||||
scope="global"
|
||||
class="inline-flex items-center gap-x-1"
|
||||
>
|
||||
<template #arrow>
|
||||
<ArrowTopRightOnSquareIcon class="size-4" />
|
||||
</template>
|
||||
</i18n-t>
|
||||
</NuxtLink>
|
||||
</div>
|
||||
|
||||
<StoreView />
|
||||
@ -73,8 +90,12 @@
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { ArrowTopRightOnSquareIcon } from "@heroicons/vue/24/outline";
|
||||
|
||||
const recent = await $dropFetch("/api/v1/store/featured");
|
||||
|
||||
const user = useUser();
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
useHead({
|
||||
|
||||
12336
pnpm-lock.yaml
generated
Normal file
12336
pnpm-lock.yaml
generated
Normal file
File diff suppressed because it is too large
Load Diff
4
pnpm-workspace.yaml
Normal file
4
pnpm-workspace.yaml
Normal file
@ -0,0 +1,4 @@
|
||||
overrides:
|
||||
droplet: link:../../.local/share/pnpm/global/5/node_modules/@drop-oss/droplet
|
||||
|
||||
shamefullyHoist: true
|
||||
@ -0,0 +1,15 @@
|
||||
/*
|
||||
Warnings:
|
||||
|
||||
- The primary key for the `Task` table will be changed. If it partially fails, the table could be left without primary key constraint.
|
||||
|
||||
*/
|
||||
-- DropIndex
|
||||
DROP INDEX "GameTag_name_idx";
|
||||
|
||||
-- AlterTable
|
||||
ALTER TABLE "Task" DROP CONSTRAINT "Task_pkey",
|
||||
ADD CONSTRAINT "Task_pkey" PRIMARY KEY ("id", "started");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "GameTag_name_idx" ON "GameTag" USING GIST ("name" gist_trgm_ops(siglen=32));
|
||||
@ -0,0 +1,8 @@
|
||||
-- DropIndex
|
||||
DROP INDEX "GameTag_name_idx";
|
||||
|
||||
-- AlterTable
|
||||
ALTER TABLE "APIToken" ADD COLUMN "expiresAt" TIMESTAMP(3);
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "GameTag_name_idx" ON "GameTag" USING GIST ("name" gist_trgm_ops(siglen=32));
|
||||
@ -0,0 +1,8 @@
|
||||
-- AlterEnum
|
||||
ALTER TYPE "MetadataSource" ADD VALUE 'Steam';
|
||||
|
||||
-- DropIndex
|
||||
DROP INDEX "GameTag_name_idx";
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "GameTag_name_idx" ON "GameTag" USING GIST ("name" gist_trgm_ops(siglen=32));
|
||||
@ -45,6 +45,8 @@ model APIToken {
|
||||
|
||||
acls String[]
|
||||
|
||||
expiresAt DateTime?
|
||||
|
||||
@@index([token])
|
||||
}
|
||||
|
||||
|
||||
@ -1,6 +1,7 @@
|
||||
enum MetadataSource {
|
||||
Manual
|
||||
GiantBomb
|
||||
Steam
|
||||
PCGamingWiki
|
||||
IGDB
|
||||
Metacritic
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
model Task {
|
||||
id String @id
|
||||
id String
|
||||
taskGroup String
|
||||
name String
|
||||
|
||||
@ -12,4 +12,6 @@ model Task {
|
||||
log String[]
|
||||
|
||||
acls String[]
|
||||
|
||||
@@id([id, started])
|
||||
}
|
||||
|
||||
@ -1 +1,2 @@
|
||||
|
||||
User-agent: *
|
||||
Disallow: /
|
||||
|
||||
47
server/api/v1/admin/company/index.post.ts
Normal file
47
server/api/v1/admin/company/index.post.ts
Normal file
@ -0,0 +1,47 @@
|
||||
import { type } from "arktype";
|
||||
import { readDropValidatedBody, throwingArktype } from "~/server/arktype";
|
||||
import aclManager from "~/server/internal/acls";
|
||||
import * as jdenticon from "jdenticon";
|
||||
import { ObjectTransactionalHandler } from "~/server/internal/objects/transactional";
|
||||
import prisma from "~/server/internal/db/database";
|
||||
import { MetadataSource } from "~/prisma/client/enums";
|
||||
|
||||
const CompanyCreate = type({
|
||||
name: "string",
|
||||
description: "string",
|
||||
website: "string",
|
||||
}).configure(throwingArktype);
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const allowed = await aclManager.allowSystemACL(h3, ["company:create"]);
|
||||
if (!allowed) throw createError({ statusCode: 403 });
|
||||
|
||||
const body = await readDropValidatedBody(h3, CompanyCreate);
|
||||
const obj = new ObjectTransactionalHandler();
|
||||
const [register, pull, _] = obj.new({}, ["internal:read"]);
|
||||
|
||||
const icon = jdenticon.toPng(body.name, 512);
|
||||
const logoId = register(icon);
|
||||
|
||||
const banner = jdenticon.toPng(body.description, 1024);
|
||||
const bannerId = register(banner);
|
||||
|
||||
const company = await prisma.company.create({
|
||||
data: {
|
||||
metadataSource: MetadataSource.Manual,
|
||||
metadataId: crypto.randomUUID(),
|
||||
metadataOriginalQuery: "",
|
||||
|
||||
mName: body.name,
|
||||
mShortDescription: body.description,
|
||||
mDescription: "",
|
||||
mLogoObjectId: logoId,
|
||||
mBannerObjectId: bannerId,
|
||||
mWebsite: body.website,
|
||||
},
|
||||
});
|
||||
|
||||
await pull();
|
||||
|
||||
return company;
|
||||
});
|
||||
@ -17,11 +17,8 @@ export default defineEventHandler(async (h3) => {
|
||||
orderBy: {
|
||||
versionIndex: "asc",
|
||||
},
|
||||
select: {
|
||||
versionIndex: true,
|
||||
versionName: true,
|
||||
platform: true,
|
||||
delta: true,
|
||||
omit: {
|
||||
dropletManifest: true,
|
||||
},
|
||||
},
|
||||
tags: true,
|
||||
|
||||
@ -18,30 +18,55 @@ export default defineEventHandler<{ body: typeof UpdateVersionOrder }>(
|
||||
const body = await readDropValidatedBody(h3, UpdateVersionOrder);
|
||||
const gameId = body.id;
|
||||
// We expect an array of the version names for this game
|
||||
const versions = body.versions;
|
||||
const unsortedVersions = await prisma.gameVersion.findMany({
|
||||
where: {
|
||||
versionName: { in: body.versions },
|
||||
},
|
||||
select: {
|
||||
versionName: true,
|
||||
versionIndex: true,
|
||||
delta: true,
|
||||
platform: true,
|
||||
},
|
||||
});
|
||||
|
||||
const newVersions = await prisma.$transaction(
|
||||
versions.map((versionName, versionIndex) =>
|
||||
const versions = body.versions
|
||||
.map((e) => unsortedVersions.find((v) => v.versionName === e))
|
||||
.filter((e) => e !== undefined);
|
||||
|
||||
if (versions.length !== unsortedVersions.length)
|
||||
throw createError({
|
||||
statusCode: 500,
|
||||
statusMessage: "Sorting versions yielded less results, somehow.",
|
||||
});
|
||||
|
||||
// Validate the new order
|
||||
const has: { [key: string]: boolean } = {};
|
||||
for (const version of versions) {
|
||||
if (version.delta && !has[version.platform])
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: `"${version.versionName}" requires a base version to apply the delta to.`,
|
||||
});
|
||||
has[version.platform] = true;
|
||||
}
|
||||
|
||||
await prisma.$transaction(
|
||||
versions.map((version, versionIndex) =>
|
||||
prisma.gameVersion.update({
|
||||
where: {
|
||||
gameId_versionName: {
|
||||
gameId: gameId,
|
||||
versionName: versionName,
|
||||
versionName: version.versionName,
|
||||
},
|
||||
},
|
||||
data: {
|
||||
versionIndex: versionIndex,
|
||||
},
|
||||
select: {
|
||||
versionIndex: true,
|
||||
versionName: true,
|
||||
platform: true,
|
||||
delta: true,
|
||||
},
|
||||
}),
|
||||
),
|
||||
);
|
||||
|
||||
return newVersions;
|
||||
return versions;
|
||||
},
|
||||
);
|
||||
|
||||
@ -7,8 +7,9 @@ export default defineEventHandler(async (h3) => {
|
||||
|
||||
const unimportedGames = await libraryManager.fetchUnimportedGames();
|
||||
const games = await libraryManager.fetchGamesWithStatus();
|
||||
const libraries = await libraryManager.fetchLibraries();
|
||||
|
||||
// Fetch other library data here
|
||||
|
||||
return { unimportedGames, games };
|
||||
return { unimportedGames, games, hasLibraries: libraries.length > 0 };
|
||||
});
|
||||
|
||||
@ -1,5 +1,6 @@
|
||||
import aclManager from "~/server/internal/acls";
|
||||
import prisma from "~/server/internal/db/database";
|
||||
import type { TaskMessage } from "~/server/internal/tasks";
|
||||
import taskHandler from "~/server/internal/tasks";
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
@ -13,7 +14,7 @@ export default defineEventHandler(async (h3) => {
|
||||
});
|
||||
|
||||
const runningTasks = (await taskHandler.runningTasks()).map((e) => e.id);
|
||||
const historicalTasks = await prisma.task.findMany({
|
||||
const historicalTasks = (await prisma.task.findMany({
|
||||
where: {
|
||||
OR: [
|
||||
{
|
||||
@ -28,7 +29,7 @@ export default defineEventHandler(async (h3) => {
|
||||
ended: "desc",
|
||||
},
|
||||
take: 10,
|
||||
});
|
||||
})) as Array<TaskMessage>;
|
||||
const dailyTasks = await taskHandler.dailyTasks();
|
||||
const weeklyTasks = await taskHandler.weeklyTasks();
|
||||
|
||||
|
||||
31
server/api/v1/admin/task/index.post.ts
Normal file
31
server/api/v1/admin/task/index.post.ts
Normal file
@ -0,0 +1,31 @@
|
||||
import { type } from "arktype";
|
||||
import { readDropValidatedBody, throwingArktype } from "~/server/arktype";
|
||||
import aclManager from "~/server/internal/acls";
|
||||
import taskHandler from "~/server/internal/tasks";
|
||||
import type { TaskGroup } from "~/server/internal/tasks/group";
|
||||
import { taskGroups } from "~/server/internal/tasks/group";
|
||||
|
||||
const StartTask = type({
|
||||
taskGroup: type("string"),
|
||||
}).configure(throwingArktype);
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const allowed = await aclManager.allowSystemACL(h3, ["task:start"]);
|
||||
if (!allowed) throw createError({ statusCode: 403 });
|
||||
|
||||
const body = await readDropValidatedBody(h3, StartTask);
|
||||
const taskGroup = body.taskGroup as TaskGroup;
|
||||
if (!taskGroups[taskGroup])
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "Invalid task group.",
|
||||
});
|
||||
|
||||
const task = await taskHandler.runTaskGroupByName(taskGroup);
|
||||
if (!task)
|
||||
throw createError({
|
||||
statusCode: 500,
|
||||
statusMessage: "Could not start task.",
|
||||
});
|
||||
return { id: task };
|
||||
});
|
||||
23
server/api/v1/admin/token/[id]/index.delete.ts
Normal file
23
server/api/v1/admin/token/[id]/index.delete.ts
Normal file
@ -0,0 +1,23 @@
|
||||
import { APITokenMode } from "~/prisma/client/enums";
|
||||
import aclManager from "~/server/internal/acls";
|
||||
import prisma from "~/server/internal/db/database";
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const allowed = await aclManager.allowSystemACL(h3, []); // No ACLs only allows session authentication
|
||||
if (!allowed) throw createError({ statusCode: 403 });
|
||||
|
||||
const id = h3.context.params?.id;
|
||||
if (!id)
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "No id in router params",
|
||||
});
|
||||
|
||||
const deleted = await prisma.aPIToken.delete({
|
||||
where: { id: id, mode: APITokenMode.System },
|
||||
})!;
|
||||
if (!deleted)
|
||||
throw createError({ statusCode: 404, statusMessage: "Token not found" });
|
||||
|
||||
return;
|
||||
});
|
||||
9
server/api/v1/admin/token/acls.get.ts
Normal file
9
server/api/v1/admin/token/acls.get.ts
Normal file
@ -0,0 +1,9 @@
|
||||
import aclManager from "~/server/internal/acls";
|
||||
import { systemACLDescriptions } from "~/server/internal/acls/descriptions";
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const allowed = await aclManager.allowSystemACL(h3, []); // No ACLs only allows session authentication
|
||||
if (!allowed) throw createError({ statusCode: 403 });
|
||||
|
||||
return systemACLDescriptions;
|
||||
});
|
||||
15
server/api/v1/admin/token/index.get.ts
Normal file
15
server/api/v1/admin/token/index.get.ts
Normal file
@ -0,0 +1,15 @@
|
||||
import { APITokenMode } from "~/prisma/client/enums";
|
||||
import aclManager from "~/server/internal/acls";
|
||||
import prisma from "~/server/internal/db/database";
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const allowed = await aclManager.allowSystemACL(h3, []); // No ACLs only allows session authentication
|
||||
if (!allowed) throw createError({ statusCode: 403 });
|
||||
|
||||
const tokens = await prisma.aPIToken.findMany({
|
||||
where: { mode: APITokenMode.System },
|
||||
omit: { token: true },
|
||||
});
|
||||
|
||||
return tokens;
|
||||
});
|
||||
38
server/api/v1/admin/token/index.post.ts
Normal file
38
server/api/v1/admin/token/index.post.ts
Normal file
@ -0,0 +1,38 @@
|
||||
import { type } from "arktype";
|
||||
import { APITokenMode } from "~/prisma/client/enums";
|
||||
import { readDropValidatedBody, throwingArktype } from "~/server/arktype";
|
||||
import aclManager, { systemACLs } from "~/server/internal/acls";
|
||||
import prisma from "~/server/internal/db/database";
|
||||
|
||||
const CreateToken = type({
|
||||
name: "string",
|
||||
acls: "string[] > 0",
|
||||
expiry: "string.date.iso.parse?",
|
||||
}).configure(throwingArktype);
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const allowed = await aclManager.allowSystemACL(h3, []); // No ACLs only allows session authentication
|
||||
if (!allowed) throw createError({ statusCode: 403 });
|
||||
|
||||
const body = await readDropValidatedBody(h3, CreateToken);
|
||||
|
||||
const invalidACLs = body.acls.filter(
|
||||
(e) => systemACLs.findIndex((v) => e == v) == -1,
|
||||
);
|
||||
if (invalidACLs.length > 0)
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: `Invalid ACLs: ${invalidACLs.join(", ")}`,
|
||||
});
|
||||
|
||||
const token = await prisma.aPIToken.create({
|
||||
data: {
|
||||
mode: APITokenMode.System,
|
||||
name: body.name,
|
||||
acls: body.acls,
|
||||
expiresAt: body.expiry ?? null,
|
||||
},
|
||||
});
|
||||
|
||||
return token;
|
||||
});
|
||||
@ -5,5 +5,6 @@ export default defineEventHandler((_h3) => {
|
||||
appName: "Drop",
|
||||
version: systemConfig.getDropVersion(),
|
||||
gitRef: `#${systemConfig.getGitRef()}`,
|
||||
external: systemConfig.getExternalUrl(),
|
||||
};
|
||||
});
|
||||
|
||||
@ -18,7 +18,8 @@ const StoreRead = type({
|
||||
company: "string?",
|
||||
companyActions: "string = 'published,developed'",
|
||||
|
||||
sort: "'default' | 'newest' | 'recent' = 'default'",
|
||||
sort: "'default' | 'newest' | 'recent' | 'name' = 'default'",
|
||||
order: "'asc' | 'desc' = 'desc'",
|
||||
});
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
@ -101,10 +102,13 @@ export default defineEventHandler(async (h3) => {
|
||||
switch (options.sort) {
|
||||
case "default":
|
||||
case "newest":
|
||||
sort.mReleased = "desc";
|
||||
sort.mReleased = options.order;
|
||||
break;
|
||||
case "recent":
|
||||
sort.created = "desc";
|
||||
sort.created = options.order;
|
||||
break;
|
||||
case "name":
|
||||
sort.mName = options.order;
|
||||
break;
|
||||
}
|
||||
|
||||
@ -119,4 +123,4 @@ export default defineEventHandler(async (h3) => {
|
||||
]);
|
||||
|
||||
return { results, count };
|
||||
});
|
||||
});
|
||||
6
server/api/v1/token.get.ts
Normal file
6
server/api/v1/token.get.ts
Normal file
@ -0,0 +1,6 @@
|
||||
import aclManager from "~/server/internal/acls";
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const acls = await aclManager.fetchAllACLs(h3);
|
||||
return acls;
|
||||
});
|
||||
@ -1,30 +1,22 @@
|
||||
import { type } from "arktype";
|
||||
import { APITokenMode } from "~/prisma/client/enums";
|
||||
import { readDropValidatedBody, throwingArktype } from "~/server/arktype";
|
||||
import aclManager, { userACLs } from "~/server/internal/acls";
|
||||
import prisma from "~/server/internal/db/database";
|
||||
|
||||
const CreateToken = type({
|
||||
name: "string",
|
||||
acls: "string[] > 0",
|
||||
expiry: "string.date.iso.parse?",
|
||||
}).configure(throwingArktype);
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const userId = await aclManager.getUserIdACL(h3, []); // No ACLs only allows session authentication
|
||||
if (!userId) throw createError({ statusCode: 403 });
|
||||
|
||||
const body = await readBody(h3);
|
||||
const name: string = body.name;
|
||||
const acls: string[] = body.acls;
|
||||
const body = await readDropValidatedBody(h3, CreateToken);
|
||||
|
||||
if (!name || typeof name !== "string")
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "Token name required",
|
||||
});
|
||||
if (!acls || !Array.isArray(acls))
|
||||
throw createError({ statusCode: 400, statusMessage: "ACLs required" });
|
||||
|
||||
if (acls.length == 0)
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "Token requires more than zero ACLs",
|
||||
});
|
||||
|
||||
const invalidACLs = acls.filter(
|
||||
const invalidACLs = body.acls.filter(
|
||||
(e) => userACLs.findIndex((v) => e == v) == -1,
|
||||
);
|
||||
if (invalidACLs.length > 0)
|
||||
@ -36,9 +28,10 @@ export default defineEventHandler(async (h3) => {
|
||||
const token = await prisma.aPIToken.create({
|
||||
data: {
|
||||
mode: APITokenMode.User,
|
||||
name: name,
|
||||
name: body.name,
|
||||
userId: userId,
|
||||
acls: acls,
|
||||
acls: body.acls,
|
||||
expiresAt: body.expiry ?? null,
|
||||
},
|
||||
});
|
||||
|
||||
|
||||
95
server/api/v2/client/chunk.post.ts
Normal file
95
server/api/v2/client/chunk.post.ts
Normal file
@ -0,0 +1,95 @@
|
||||
import { type } from "arktype";
|
||||
import { readDropValidatedBody, throwingArktype } from "~/server/arktype";
|
||||
import contextManager from "~/server/internal/downloads/coordinator";
|
||||
import libraryManager from "~/server/internal/library";
|
||||
import { logger } from "~/server/internal/logging";
|
||||
|
||||
const GetChunk = type({
|
||||
context: "string",
|
||||
files: type({
|
||||
filename: "string",
|
||||
chunkIndex: "number",
|
||||
})
|
||||
.array()
|
||||
.atLeastLength(1)
|
||||
.atMostLength(256),
|
||||
}).configure(throwingArktype);
|
||||
|
||||
export default defineEventHandler(async (h3) => {
|
||||
const body = await readDropValidatedBody(h3, GetChunk);
|
||||
|
||||
const context = await contextManager.fetchContext(body.context);
|
||||
if (!context)
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "Invalid download context.",
|
||||
});
|
||||
|
||||
const streamFiles = [];
|
||||
|
||||
for (const file of body.files) {
|
||||
const manifestFile = context.manifest[file.filename];
|
||||
if (!manifestFile)
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: `Unknown file: ${file.filename}`,
|
||||
});
|
||||
|
||||
const start = manifestFile.lengths
|
||||
.slice(0, file.chunkIndex)
|
||||
.reduce((a, b) => a + b, 0);
|
||||
const end = start + manifestFile.lengths[file.chunkIndex];
|
||||
|
||||
streamFiles.push({ filename: file.filename, start, end });
|
||||
}
|
||||
|
||||
setHeader(
|
||||
h3,
|
||||
"Content-Lengths",
|
||||
streamFiles.map((e) => e.end - e.start).join(","),
|
||||
); // Non-standard header, but we're cool like that 😎
|
||||
|
||||
const streams = await Promise.all(
|
||||
streamFiles.map(async (file) => {
|
||||
const gameReadStream = await libraryManager.readFile(
|
||||
context.libraryId,
|
||||
context.libraryPath,
|
||||
context.versionName,
|
||||
file.filename,
|
||||
{ start: file.start, end: file.end },
|
||||
);
|
||||
if (!gameReadStream)
|
||||
throw createError({
|
||||
statusCode: 500,
|
||||
statusMessage: "Failed to create read stream",
|
||||
});
|
||||
return { ...file, stream: gameReadStream };
|
||||
}),
|
||||
);
|
||||
|
||||
for (const file of streams) {
|
||||
let length = 0;
|
||||
await file.stream.pipeTo(
|
||||
new WritableStream({
|
||||
write(chunk) {
|
||||
h3.node.res.write(chunk);
|
||||
length += chunk.length;
|
||||
},
|
||||
}),
|
||||
);
|
||||
|
||||
if (length != file.end - file.start) {
|
||||
logger.warn(
|
||||
`failed to read enough from ${file.filename}. read ${length}, required: ${file.end - file.start}`,
|
||||
);
|
||||
throw createError({
|
||||
statusCode: 500,
|
||||
statusMessage: "Failed to read enough from stream.",
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
await h3.node.res.end();
|
||||
|
||||
return;
|
||||
});
|
||||
22
server/api/v2/client/context.post.ts
Normal file
22
server/api/v2/client/context.post.ts
Normal file
@ -0,0 +1,22 @@
|
||||
import { type } from "arktype";
|
||||
import { readDropValidatedBody, throwingArktype } from "~/server/arktype";
|
||||
import { defineClientEventHandler } from "~/server/internal/clients/event-handler";
|
||||
import contextManager from "~/server/internal/downloads/coordinator";
|
||||
|
||||
const CreateContext = type({
|
||||
game: "string",
|
||||
version: "string",
|
||||
}).configure(throwingArktype);
|
||||
|
||||
export default defineClientEventHandler(async (h3) => {
|
||||
const body = await readDropValidatedBody(h3, CreateContext);
|
||||
|
||||
const context = await contextManager.createContext(body.game, body.version);
|
||||
if (!context)
|
||||
throw createError({
|
||||
statusCode: 400,
|
||||
statusMessage: "Invalid game or version",
|
||||
});
|
||||
|
||||
return { context };
|
||||
});
|
||||
@ -19,7 +19,7 @@ export const userACLDescriptions: ObjectFromList<typeof userACLs> = {
|
||||
|
||||
"notifications:read": "Fetch this account's notifications.",
|
||||
"notifications:mark": "Mark notifications as read for this account.",
|
||||
"notifications:listen": "Connect to a websocket to recieve notifications.",
|
||||
"notifications:listen": "Connect to a websocket to receive notifications.",
|
||||
"notifications:delete": "Delete this account's notifications.",
|
||||
|
||||
"screenshots:new": "Create screenshots for this account",
|
||||
@ -36,7 +36,7 @@ export const userACLDescriptions: ObjectFromList<typeof userACLs> = {
|
||||
"library:remove": "Remove a game from your library.",
|
||||
|
||||
"clients:read": "Read the clients connected to this account",
|
||||
"clients:revoke": "",
|
||||
"clients:revoke": "Remove clients connected to this account",
|
||||
|
||||
"news:read": "Read the server's news articles.",
|
||||
|
||||
|
||||
@ -16,7 +16,7 @@ Server sends redirect to `drop://handshake/[id]/[token]`, where the token is an
|
||||
|
||||
## 3. Client requests certificates
|
||||
|
||||
Client makes request: `POST /api/v1/client/auth/handshake` with the token recieved in the previous step.
|
||||
Client makes request: `POST /api/v1/client/auth/handshake` with the token received in the previous step.
|
||||
|
||||
The server uses it's CA to generate a public-private key pair, the CN of the client ID. It then sends that pair, plus the CA's public key, to the client, which stores it all.
|
||||
|
||||
|
||||
@ -1,9 +1,68 @@
|
||||
/*
|
||||
The download co-ordinator's job is to keep track of all the currently online clients.
|
||||
import prisma from "../db/database";
|
||||
import type { DropManifest } from "./manifest";
|
||||
|
||||
When a client signs on and registers itself as a peer
|
||||
const TIMEOUT = 1000 * 60 * 60 * 1; // 1 hour
|
||||
|
||||
*/
|
||||
class DownloadContextManager {
|
||||
private contexts: Map<
|
||||
string,
|
||||
{
|
||||
timeout: Date;
|
||||
manifest: DropManifest;
|
||||
versionName: string;
|
||||
libraryId: string;
|
||||
libraryPath: string;
|
||||
}
|
||||
> = new Map();
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-extraneous-class, @typescript-eslint/no-unused-vars
|
||||
class DownloadCoordinator {}
|
||||
async createContext(game: string, versionName: string) {
|
||||
const version = await prisma.gameVersion.findUnique({
|
||||
where: {
|
||||
gameId_versionName: {
|
||||
gameId: game,
|
||||
versionName,
|
||||
},
|
||||
},
|
||||
include: {
|
||||
game: {
|
||||
select: {
|
||||
libraryId: true,
|
||||
libraryPath: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
if (!version) return undefined;
|
||||
|
||||
const contextId = crypto.randomUUID();
|
||||
this.contexts.set(contextId, {
|
||||
timeout: new Date(),
|
||||
manifest: JSON.parse(version.dropletManifest as string) as DropManifest,
|
||||
versionName,
|
||||
libraryId: version.game.libraryId!,
|
||||
libraryPath: version.game.libraryPath,
|
||||
});
|
||||
|
||||
return contextId;
|
||||
}
|
||||
|
||||
async fetchContext(contextId: string) {
|
||||
const context = this.contexts.get(contextId);
|
||||
if (!context) return undefined;
|
||||
context.timeout = new Date();
|
||||
this.contexts.set(contextId, context);
|
||||
return context;
|
||||
}
|
||||
|
||||
async cleanup() {
|
||||
for (const key of this.contexts.keys()) {
|
||||
const context = this.contexts.get(key)!;
|
||||
if (context.timeout.getTime() < Date.now() - TIMEOUT) {
|
||||
this.contexts.delete(key);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export const contextManager = new DownloadContextManager();
|
||||
export default contextManager;
|
||||
|
||||
@ -5,7 +5,7 @@ export type DropChunk = {
|
||||
permissions: number;
|
||||
ids: string[];
|
||||
checksums: string[];
|
||||
lengths: string[];
|
||||
lengths: number[];
|
||||
};
|
||||
|
||||
export type DropManifest = {
|
||||
|
||||
@ -13,13 +13,24 @@ import { parsePlatform } from "../utils/parseplatform";
|
||||
import notificationSystem from "../notifications";
|
||||
import { GameNotFoundError, type LibraryProvider } from "./provider";
|
||||
import { logger } from "../logging";
|
||||
import type { GameModel } from "~/prisma/client/models";
|
||||
import { createHash } from "node:crypto";
|
||||
|
||||
export function createGameImportTaskId(libraryId: string, libraryPath: string) {
|
||||
return createHash("md5")
|
||||
.update(`import:${libraryId}:${libraryPath}`)
|
||||
.digest("hex");
|
||||
}
|
||||
|
||||
export function createVersionImportTaskId(gameId: string, versionName: string) {
|
||||
return createHash("md5")
|
||||
.update(`import:${gameId}:${versionName}`)
|
||||
.digest("hex");
|
||||
}
|
||||
|
||||
class LibraryManager {
|
||||
private libraries: Map<string, LibraryProvider<unknown>> = new Map();
|
||||
|
||||
private gameImportLocks: Map<string, Array<string>> = new Map(); // Library ID to Library Path
|
||||
private versionImportLocks: Map<string, Array<string>> = new Map(); // Game ID to Version Name
|
||||
|
||||
addLibrary(library: LibraryProvider<unknown>) {
|
||||
this.libraries.set(library.id(), library);
|
||||
}
|
||||
@ -37,24 +48,30 @@ class LibraryManager {
|
||||
return libraryWithMetadata;
|
||||
}
|
||||
|
||||
async fetchGamesByLibrary() {
|
||||
const results: { [key: string]: { [key: string]: GameModel } } = {};
|
||||
const games = await prisma.game.findMany({});
|
||||
for (const game of games) {
|
||||
const libraryId = game.libraryId!;
|
||||
const libraryPath = game.libraryPath!;
|
||||
|
||||
results[libraryId] ??= {};
|
||||
results[libraryId][libraryPath] = game;
|
||||
}
|
||||
|
||||
return results;
|
||||
}
|
||||
|
||||
async fetchUnimportedGames() {
|
||||
const unimportedGames: { [key: string]: string[] } = {};
|
||||
const instanceGames = await this.fetchGamesByLibrary();
|
||||
|
||||
for (const [id, library] of this.libraries.entries()) {
|
||||
const games = await library.listGames();
|
||||
const validGames = await prisma.game.findMany({
|
||||
where: {
|
||||
libraryId: id,
|
||||
libraryPath: { in: games },
|
||||
},
|
||||
select: {
|
||||
libraryPath: true,
|
||||
},
|
||||
});
|
||||
const providerUnimportedGames = games.filter(
|
||||
(e) =>
|
||||
validGames.findIndex((v) => v.libraryPath == e) == -1 &&
|
||||
!(this.gameImportLocks.get(id) ?? []).includes(e),
|
||||
const providerGames = await library.listGames();
|
||||
const providerUnimportedGames = providerGames.filter(
|
||||
(libraryPath) =>
|
||||
!instanceGames[id]?.[libraryPath] &&
|
||||
!taskHandler.hasTask(createGameImportTaskId(id, libraryPath)),
|
||||
);
|
||||
unimportedGames[id] = providerUnimportedGames;
|
||||
}
|
||||
@ -84,7 +101,7 @@ class LibraryManager {
|
||||
const unimportedVersions = versions.filter(
|
||||
(e) =>
|
||||
game.versions.findIndex((v) => v.versionName == e) == -1 &&
|
||||
!(this.versionImportLocks.get(game.id) ?? []).includes(e),
|
||||
!taskHandler.hasTask(createVersionImportTaskId(game.id, e)),
|
||||
);
|
||||
return unimportedVersions;
|
||||
} catch (e) {
|
||||
@ -99,7 +116,11 @@ class LibraryManager {
|
||||
async fetchGamesWithStatus() {
|
||||
const games = await prisma.game.findMany({
|
||||
include: {
|
||||
versions: true,
|
||||
versions: {
|
||||
select: {
|
||||
versionName: true,
|
||||
},
|
||||
},
|
||||
library: true,
|
||||
},
|
||||
orderBy: {
|
||||
@ -150,6 +171,8 @@ class LibraryManager {
|
||||
".sh",
|
||||
// No extension is common for Linux binaries
|
||||
"",
|
||||
// AppImages
|
||||
".appimage",
|
||||
],
|
||||
Windows: [".exe", ".bat"],
|
||||
macOS: [
|
||||
@ -168,7 +191,8 @@ class LibraryManager {
|
||||
for (const filename of files) {
|
||||
const basename = path.basename(filename);
|
||||
const dotLocation = filename.lastIndexOf(".");
|
||||
const ext = dotLocation == -1 ? "" : filename.slice(dotLocation);
|
||||
const ext =
|
||||
dotLocation == -1 ? "" : filename.slice(dotLocation).toLowerCase();
|
||||
for (const [platform, checkExts] of Object.entries(fileExts)) {
|
||||
for (const checkExt of checkExts) {
|
||||
if (checkExt != ext) continue;
|
||||
@ -206,70 +230,6 @@ class LibraryManager {
|
||||
}
|
||||
*/
|
||||
|
||||
/**
|
||||
* Locks the game so you can't be imported
|
||||
* @param libraryId
|
||||
* @param libraryPath
|
||||
*/
|
||||
async lockGame(libraryId: string, libraryPath: string) {
|
||||
let games = this.gameImportLocks.get(libraryId);
|
||||
if (!games) this.gameImportLocks.set(libraryId, (games = []));
|
||||
|
||||
if (!games.includes(libraryPath)) games.push(libraryPath);
|
||||
|
||||
this.gameImportLocks.set(libraryId, games);
|
||||
}
|
||||
|
||||
/**
|
||||
* Unlocks the game, call once imported
|
||||
* @param libraryId
|
||||
* @param libraryPath
|
||||
*/
|
||||
async unlockGame(libraryId: string, libraryPath: string) {
|
||||
let games = this.gameImportLocks.get(libraryId);
|
||||
if (!games) this.gameImportLocks.set(libraryId, (games = []));
|
||||
|
||||
if (games.includes(libraryPath))
|
||||
games.splice(
|
||||
games.findIndex((e) => e === libraryPath),
|
||||
1,
|
||||
);
|
||||
|
||||
this.gameImportLocks.set(libraryId, games);
|
||||
}
|
||||
|
||||
/**
|
||||
* Locks a version so it can't be imported
|
||||
* @param gameId
|
||||
* @param versionName
|
||||
*/
|
||||
async lockVersion(gameId: string, versionName: string) {
|
||||
let versions = this.versionImportLocks.get(gameId);
|
||||
if (!versions) this.versionImportLocks.set(gameId, (versions = []));
|
||||
|
||||
if (!versions.includes(versionName)) versions.push(versionName);
|
||||
|
||||
this.versionImportLocks.set(gameId, versions);
|
||||
}
|
||||
|
||||
/**
|
||||
* Unlocks the version, call once imported
|
||||
* @param libraryId
|
||||
* @param libraryPath
|
||||
*/
|
||||
async unlockVersion(gameId: string, versionName: string) {
|
||||
let versions = this.versionImportLocks.get(gameId);
|
||||
if (!versions) this.versionImportLocks.set(gameId, (versions = []));
|
||||
|
||||
if (versions.includes(gameId))
|
||||
versions.splice(
|
||||
versions.findIndex((e) => e === versionName),
|
||||
1,
|
||||
);
|
||||
|
||||
this.versionImportLocks.set(gameId, versions);
|
||||
}
|
||||
|
||||
async importVersion(
|
||||
gameId: string,
|
||||
versionName: string,
|
||||
@ -286,7 +246,7 @@ class LibraryManager {
|
||||
umuId: string;
|
||||
},
|
||||
) {
|
||||
const taskId = `import:${gameId}:${versionName}`;
|
||||
const taskId = createVersionImportTaskId(gameId, versionName);
|
||||
|
||||
const platform = parsePlatform(metadata.platform);
|
||||
if (!platform) return undefined;
|
||||
@ -300,8 +260,6 @@ class LibraryManager {
|
||||
const library = this.libraries.get(game.libraryId);
|
||||
if (!library) return undefined;
|
||||
|
||||
await this.lockVersion(gameId, versionName);
|
||||
|
||||
taskHandler.create({
|
||||
id: taskId,
|
||||
taskGroup: "import:game",
|
||||
@ -378,9 +336,6 @@ class LibraryManager {
|
||||
|
||||
progress(100);
|
||||
},
|
||||
async finally() {
|
||||
await libraryManager.unlockVersion(gameId, versionName);
|
||||
},
|
||||
});
|
||||
|
||||
return taskId;
|
||||
@ -394,7 +349,7 @@ class LibraryManager {
|
||||
) {
|
||||
const library = this.libraries.get(libraryId);
|
||||
if (!library) return undefined;
|
||||
return library.peekFile(game, version, filename);
|
||||
return await library.peekFile(game, version, filename);
|
||||
}
|
||||
|
||||
async readFile(
|
||||
@ -406,7 +361,7 @@ class LibraryManager {
|
||||
) {
|
||||
const library = this.libraries.get(libraryId);
|
||||
if (!library) return undefined;
|
||||
return library.readFile(game, version, filename, options);
|
||||
return await library.readFile(game, version, filename, options);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@ -7,12 +7,14 @@ import {
|
||||
import { LibraryBackend } from "~/prisma/client/enums";
|
||||
import fs from "fs";
|
||||
import path from "path";
|
||||
import droplet from "@drop-oss/droplet";
|
||||
import droplet, { DropletHandler } from "@drop-oss/droplet";
|
||||
|
||||
export const FilesystemProviderConfig = type({
|
||||
baseDir: "string",
|
||||
});
|
||||
|
||||
export const DROPLET_HANDLER = new DropletHandler();
|
||||
|
||||
export class FilesystemProvider
|
||||
implements LibraryProvider<typeof FilesystemProviderConfig.infer>
|
||||
{
|
||||
@ -57,7 +59,7 @@ export class FilesystemProvider
|
||||
const versionDirs = fs.readdirSync(gameDir);
|
||||
const validVersionDirs = versionDirs.filter((e) => {
|
||||
const fullDir = path.join(this.config.baseDir, game, e);
|
||||
return droplet.hasBackendForPath(fullDir);
|
||||
return DROPLET_HANDLER.hasBackendForPath(fullDir);
|
||||
});
|
||||
return validVersionDirs;
|
||||
}
|
||||
@ -65,7 +67,7 @@ export class FilesystemProvider
|
||||
async versionReaddir(game: string, version: string): Promise<string[]> {
|
||||
const versionDir = path.join(this.config.baseDir, game, version);
|
||||
if (!fs.existsSync(versionDir)) throw new VersionNotFoundError();
|
||||
return droplet.listFiles(versionDir);
|
||||
return DROPLET_HANDLER.listFiles(versionDir);
|
||||
}
|
||||
|
||||
async generateDropletManifest(
|
||||
@ -77,10 +79,16 @@ export class FilesystemProvider
|
||||
const versionDir = path.join(this.config.baseDir, game, version);
|
||||
if (!fs.existsSync(versionDir)) throw new VersionNotFoundError();
|
||||
const manifest = await new Promise<string>((r, j) =>
|
||||
droplet.generateManifest(versionDir, progress, log, (err, result) => {
|
||||
if (err) return j(err);
|
||||
r(result);
|
||||
}),
|
||||
droplet.generateManifest(
|
||||
DROPLET_HANDLER,
|
||||
versionDir,
|
||||
progress,
|
||||
log,
|
||||
(err, result) => {
|
||||
if (err) return j(err);
|
||||
r(result);
|
||||
},
|
||||
),
|
||||
);
|
||||
return manifest;
|
||||
}
|
||||
@ -88,7 +96,7 @@ export class FilesystemProvider
|
||||
async peekFile(game: string, version: string, filename: string) {
|
||||
const filepath = path.join(this.config.baseDir, game, version);
|
||||
if (!fs.existsSync(filepath)) return undefined;
|
||||
const stat = droplet.peekFile(filepath, filename);
|
||||
const stat = DROPLET_HANDLER.peekFile(filepath, filename);
|
||||
return { size: Number(stat) };
|
||||
}
|
||||
|
||||
@ -100,13 +108,17 @@ export class FilesystemProvider
|
||||
) {
|
||||
const filepath = path.join(this.config.baseDir, game, version);
|
||||
if (!fs.existsSync(filepath)) return undefined;
|
||||
const stream = droplet.readFile(
|
||||
filepath,
|
||||
filename,
|
||||
options?.start ? BigInt(options.start) : undefined,
|
||||
options?.end ? BigInt(options.end) : undefined,
|
||||
);
|
||||
if (!stream) return undefined;
|
||||
let stream;
|
||||
while (!(stream instanceof ReadableStream)) {
|
||||
const v = DROPLET_HANDLER.readFile(
|
||||
filepath,
|
||||
filename,
|
||||
options?.start ? BigInt(options.start) : undefined,
|
||||
options?.end ? BigInt(options.end) : undefined,
|
||||
);
|
||||
if (!v) return undefined;
|
||||
stream = v.getStream() as ReadableStream<unknown>;
|
||||
}
|
||||
|
||||
return stream;
|
||||
}
|
||||
|
||||
@ -5,6 +5,7 @@ import { LibraryBackend } from "~/prisma/client/enums";
|
||||
import fs from "fs";
|
||||
import path from "path";
|
||||
import droplet from "@drop-oss/droplet";
|
||||
import { DROPLET_HANDLER } from "./filesystem";
|
||||
|
||||
export const FlatFilesystemProviderConfig = type({
|
||||
baseDir: "string",
|
||||
@ -46,7 +47,7 @@ export class FlatFilesystemProvider
|
||||
const versionDirs = fs.readdirSync(this.config.baseDir);
|
||||
const validVersionDirs = versionDirs.filter((e) => {
|
||||
const fullDir = path.join(this.config.baseDir, e);
|
||||
return droplet.hasBackendForPath(fullDir);
|
||||
return DROPLET_HANDLER.hasBackendForPath(fullDir);
|
||||
});
|
||||
return validVersionDirs;
|
||||
}
|
||||
@ -63,7 +64,7 @@ export class FlatFilesystemProvider
|
||||
async versionReaddir(game: string, _version: string) {
|
||||
const versionDir = path.join(this.config.baseDir, game);
|
||||
if (!fs.existsSync(versionDir)) throw new VersionNotFoundError();
|
||||
return droplet.listFiles(versionDir);
|
||||
return DROPLET_HANDLER.listFiles(versionDir);
|
||||
}
|
||||
|
||||
async generateDropletManifest(
|
||||
@ -75,17 +76,23 @@ export class FlatFilesystemProvider
|
||||
const versionDir = path.join(this.config.baseDir, game);
|
||||
if (!fs.existsSync(versionDir)) throw new VersionNotFoundError();
|
||||
const manifest = await new Promise<string>((r, j) =>
|
||||
droplet.generateManifest(versionDir, progress, log, (err, result) => {
|
||||
if (err) return j(err);
|
||||
r(result);
|
||||
}),
|
||||
droplet.generateManifest(
|
||||
DROPLET_HANDLER,
|
||||
versionDir,
|
||||
progress,
|
||||
log,
|
||||
(err, result) => {
|
||||
if (err) return j(err);
|
||||
r(result);
|
||||
},
|
||||
),
|
||||
);
|
||||
return manifest;
|
||||
}
|
||||
async peekFile(game: string, _version: string, filename: string) {
|
||||
const filepath = path.join(this.config.baseDir, game);
|
||||
if (!fs.existsSync(filepath)) return undefined;
|
||||
const stat = droplet.peekFile(filepath, filename);
|
||||
const stat = DROPLET_HANDLER.peekFile(filepath, filename);
|
||||
return { size: Number(stat) };
|
||||
}
|
||||
async readFile(
|
||||
@ -96,7 +103,7 @@ export class FlatFilesystemProvider
|
||||
) {
|
||||
const filepath = path.join(this.config.baseDir, game);
|
||||
if (!fs.existsSync(filepath)) return undefined;
|
||||
const stream = droplet.readFile(
|
||||
const stream = DROPLET_HANDLER.readFile(
|
||||
filepath,
|
||||
filename,
|
||||
options?.start ? BigInt(options.start) : undefined,
|
||||
@ -104,6 +111,6 @@ export class FlatFilesystemProvider
|
||||
);
|
||||
if (!stream) return undefined;
|
||||
|
||||
return stream;
|
||||
return stream.getStream();
|
||||
}
|
||||
}
|
||||
|
||||
@ -191,7 +191,7 @@ export class GiantBombProvider implements MetadataProvider {
|
||||
|
||||
const res = await publisher(pub.name);
|
||||
if (res === undefined) {
|
||||
context?.logger.warn(`Failed to import publisher "${pub}"`);
|
||||
context?.logger.warn(`Failed to import publisher "${pub.name}"`);
|
||||
continue;
|
||||
}
|
||||
context?.logger.info(`Imported publisher "${pub.name}"`);
|
||||
@ -208,10 +208,10 @@ export class GiantBombProvider implements MetadataProvider {
|
||||
|
||||
const res = await developer(dev.name);
|
||||
if (res === undefined) {
|
||||
context?.logger.warn(`Failed to import developer "${dev}"`);
|
||||
context?.logger.warn(`Failed to import developer "${dev.name}"`);
|
||||
continue;
|
||||
}
|
||||
context?.logger.info(`Imported developer "${dev}"`);
|
||||
context?.logger.info(`Imported developer "${dev.name}"`);
|
||||
developers.push(res);
|
||||
}
|
||||
}
|
||||
|
||||
@ -18,7 +18,7 @@ import taskHandler, { wrapTaskContext } from "../tasks";
|
||||
import { randomUUID } from "crypto";
|
||||
import { fuzzy } from "fast-fuzzy";
|
||||
import { logger } from "~/server/internal/logging";
|
||||
import libraryManager from "../library";
|
||||
import { createGameImportTaskId } from "../library";
|
||||
import type { GameTagModel } from "~/prisma/client/models";
|
||||
|
||||
export class MissingMetadataProviderConfig extends Error {
|
||||
@ -185,11 +185,9 @@ export class MetadataHandler {
|
||||
});
|
||||
if (existing) return undefined;
|
||||
|
||||
await libraryManager.lockGame(libraryId, libraryPath);
|
||||
|
||||
const gameId = randomUUID();
|
||||
|
||||
const taskId = `import:${gameId}`;
|
||||
const taskId = createGameImportTaskId(libraryId, libraryPath);
|
||||
await taskHandler.create({
|
||||
name: `Import game "${result.name}" (${libraryPath})`,
|
||||
id: taskId,
|
||||
@ -280,9 +278,6 @@ export class MetadataHandler {
|
||||
logger.info(`Finished game import.`);
|
||||
progress(100);
|
||||
},
|
||||
async finally() {
|
||||
await libraryManager.unlockGame(libraryId, libraryPath);
|
||||
},
|
||||
});
|
||||
|
||||
return taskId;
|
||||
@ -311,7 +306,7 @@ export class MetadataHandler {
|
||||
result = await provider.fetchCompany({ query, createObject });
|
||||
if (result === undefined) {
|
||||
throw new Error(
|
||||
`${provider.source()} failed to find a company for "${query}`,
|
||||
`${provider.source()} failed to find a company for "${query}"`,
|
||||
);
|
||||
}
|
||||
} catch (e) {
|
||||
|
||||
1022
server/internal/metadata/steam.ts
Normal file
1022
server/internal/metadata/steam.ts
Normal file
File diff suppressed because it is too large
Load Diff
@ -1,5 +1,5 @@
|
||||
/*
|
||||
The notification system handles the recieving, creation and sending of notifications in Drop
|
||||
The notification system handles the receiving, creation and sending of notifications in Drop
|
||||
|
||||
Design goals:
|
||||
1. Nonce-based notifications; notifications should only be created once
|
||||
|
||||
@ -195,7 +195,7 @@ export class ObjectHandler {
|
||||
* @returns
|
||||
* @description If we need to fetch a remote resource, it doesn't make sense
|
||||
* to immediately fetch the object, *then* check permissions.
|
||||
* Instead the caller can pass a simple anonymous funciton, like
|
||||
* Instead the caller can pass a simple anonymous function, like
|
||||
* () => $dropFetch('/my-image');
|
||||
* And if we actually have permission to write, it fetches it then.
|
||||
*/
|
||||
|
||||
@ -14,6 +14,9 @@ export const taskGroups = {
|
||||
"import:game": {
|
||||
concurrency: true,
|
||||
},
|
||||
debug: {
|
||||
concurrency: true,
|
||||
},
|
||||
} as const;
|
||||
|
||||
export type TaskGroup = keyof typeof taskGroups;
|
||||
|
||||
@ -41,7 +41,7 @@ type TaskPoolEntry = FinishedTask & {
|
||||
* easily without re-inventing the wheel every time.
|
||||
*/
|
||||
class TaskHandler {
|
||||
// registry of schedualed tasks to be created
|
||||
// registry of scheduled tasks to be created
|
||||
private taskCreators: Map<TaskGroup, () => Task> = new Map();
|
||||
|
||||
// list of all currently running tasks
|
||||
@ -53,6 +53,7 @@ class TaskHandler {
|
||||
"cleanup:invitations",
|
||||
"cleanup:sessions",
|
||||
"check:update",
|
||||
"debug",
|
||||
];
|
||||
private weeklyScheduledTasks: TaskGroup[] = ["cleanup:objects"];
|
||||
|
||||
@ -62,6 +63,7 @@ class TaskHandler {
|
||||
this.saveScheduledTask(cleanupSessions);
|
||||
this.saveScheduledTask(checkUpdate);
|
||||
this.saveScheduledTask(cleanupObjects);
|
||||
//this.saveScheduledTask(debug);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -73,6 +75,8 @@ class TaskHandler {
|
||||
}
|
||||
|
||||
async create(task: Task) {
|
||||
if (this.hasTask(task.id)) throw new Error("Task with ID already exists.");
|
||||
|
||||
let updateCollectTimeout: NodeJS.Timeout | undefined;
|
||||
let updateCollectResolves: Array<(value: unknown) => void> = [];
|
||||
let logOffset: number = 0;
|
||||
@ -160,6 +164,13 @@ class TaskHandler {
|
||||
// You can configure timestamp, level, etc. here
|
||||
timestamp: pino.stdTimeFunctions.isoTime,
|
||||
base: null, // Remove pid/hostname if not needed
|
||||
formatters: {
|
||||
level(label) {
|
||||
return {
|
||||
level: label,
|
||||
};
|
||||
},
|
||||
},
|
||||
},
|
||||
logStream,
|
||||
);
|
||||
@ -206,8 +217,6 @@ class TaskHandler {
|
||||
};
|
||||
}
|
||||
|
||||
if (task.finally) await task.finally();
|
||||
|
||||
taskEntry.endTime = new Date().toISOString();
|
||||
await updateAllClients();
|
||||
|
||||
@ -247,7 +256,10 @@ class TaskHandler {
|
||||
) {
|
||||
const task =
|
||||
this.taskPool.get(taskId) ??
|
||||
(await prisma.task.findUnique({ where: { id: taskId } }));
|
||||
(await prisma.task.findFirst({
|
||||
where: { id: taskId },
|
||||
orderBy: { started: "desc" },
|
||||
}));
|
||||
if (!task) {
|
||||
peer.send(
|
||||
`error/${taskId}/Unknown task/Drop couldn't find the task you're looking for.`,
|
||||
@ -324,6 +336,10 @@ class TaskHandler {
|
||||
.toArray();
|
||||
}
|
||||
|
||||
hasTask(id: string) {
|
||||
return this.taskPool.has(id);
|
||||
}
|
||||
|
||||
dailyTasks() {
|
||||
return this.dailyScheduledTasks;
|
||||
}
|
||||
@ -332,13 +348,15 @@ class TaskHandler {
|
||||
return this.weeklyScheduledTasks;
|
||||
}
|
||||
|
||||
runTaskGroupByName(name: TaskGroup) {
|
||||
const task = this.taskCreators.get(name);
|
||||
if (!task) {
|
||||
async runTaskGroupByName(name: TaskGroup) {
|
||||
const taskConstructor = this.taskCreators.get(name);
|
||||
if (!taskConstructor) {
|
||||
logger.warn(`No task found for group ${name}`);
|
||||
return;
|
||||
}
|
||||
this.create(task());
|
||||
const task = taskConstructor();
|
||||
await this.create(task);
|
||||
return task.id;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -429,7 +447,6 @@ export interface Task {
|
||||
taskGroup: TaskGroup;
|
||||
name: string;
|
||||
run: (context: TaskRunContext) => Promise<void>;
|
||||
finally?: () => Promise<void> | void;
|
||||
acls: GlobalACL[];
|
||||
}
|
||||
|
||||
@ -438,7 +455,7 @@ export type TaskMessage = {
|
||||
name: string;
|
||||
success: boolean;
|
||||
progress: number;
|
||||
error: undefined | { title: string; description: string };
|
||||
error: null | undefined | { title: string; description: string };
|
||||
log: string[];
|
||||
reset?: boolean;
|
||||
};
|
||||
@ -464,6 +481,7 @@ interface DropTask {
|
||||
export const TaskLog = type({
|
||||
timestamp: "string",
|
||||
message: "string",
|
||||
level: "string",
|
||||
});
|
||||
|
||||
// /**
|
||||
@ -493,8 +511,6 @@ export const TaskLog = type({
|
||||
// }
|
||||
|
||||
export function defineDropTask(buildTask: BuildTask): DropTask {
|
||||
// TODO: only let one task with the same taskGroup run at the same time if specified
|
||||
|
||||
return {
|
||||
taskGroup: buildTask.taskGroup,
|
||||
build: () => ({
|
||||
|
||||
18
server/internal/tasks/registry/debug.ts
Normal file
18
server/internal/tasks/registry/debug.ts
Normal file
@ -0,0 +1,18 @@
|
||||
import { defineDropTask } from "..";
|
||||
|
||||
export default defineDropTask({
|
||||
buildId: () => `debug:${new Date().toISOString()}`,
|
||||
name: "Debug Task",
|
||||
acls: ["system:maintenance:read"],
|
||||
taskGroup: "debug",
|
||||
async run({ progress, logger }) {
|
||||
const amount = 1000;
|
||||
for (let i = 0; i < amount; i++) {
|
||||
progress((i / amount) * 100);
|
||||
logger.info(`dajksdkajd ${i}`);
|
||||
logger.warn("warning");
|
||||
logger.error("error\nmultiline and stuff\nwoah more lines");
|
||||
await new Promise((r) => setTimeout(r, 1500));
|
||||
}
|
||||
},
|
||||
});
|
||||
3
server/middleware/latency.ts
Normal file
3
server/middleware/latency.ts
Normal file
@ -0,0 +1,3 @@
|
||||
export default defineEventHandler(async () => {
|
||||
// await new Promise((r) => setTimeout(r, 1400));
|
||||
});
|
||||
@ -3,7 +3,7 @@ import prisma from "~/server/internal/db/database";
|
||||
export default defineNitroPlugin(async (_nitro) => {
|
||||
// Ensure system user exists
|
||||
// The system user owns any user-based code
|
||||
// that we want to re-use for the app
|
||||
// that we want to reuse for the app
|
||||
// e.g. notifications
|
||||
await prisma.user.upsert({
|
||||
where: {
|
||||
|
||||
@ -5,11 +5,13 @@ import { GiantBombProvider } from "../internal/metadata/giantbomb";
|
||||
import { IGDBProvider } from "../internal/metadata/igdb";
|
||||
import { ManualMetadataProvider } from "../internal/metadata/manual";
|
||||
import { PCGamingWikiProvider } from "../internal/metadata/pcgamingwiki";
|
||||
import { SteamProvider } from "../internal/metadata/steam";
|
||||
import { logger } from "~/server/internal/logging";
|
||||
|
||||
export default defineNitroPlugin(async (_nitro) => {
|
||||
const metadataProviders = [
|
||||
GiantBombProvider,
|
||||
SteamProvider,
|
||||
PCGamingWikiProvider,
|
||||
IGDBProvider,
|
||||
];
|
||||
|
||||
11
server/tasks/downloadCleanup.ts
Normal file
11
server/tasks/downloadCleanup.ts
Normal file
@ -0,0 +1,11 @@
|
||||
import contextManager from "../internal/downloads/coordinator";
|
||||
|
||||
export default defineTask({
|
||||
meta: {
|
||||
name: "downloadCleanup",
|
||||
},
|
||||
async run() {
|
||||
await contextManager.cleanup();
|
||||
return { result: true };
|
||||
},
|
||||
});
|
||||
@ -1,10 +1,31 @@
|
||||
import type { TaskLog } from "~/server/internal/tasks";
|
||||
|
||||
export function parseTaskLog(logStr: string): typeof TaskLog.infer {
|
||||
const labelNumberMap = {
|
||||
100: "silent",
|
||||
60: "fatal",
|
||||
50: "error",
|
||||
40: "warn",
|
||||
30: "info",
|
||||
20: "debug",
|
||||
10: "trace",
|
||||
0: "off",
|
||||
};
|
||||
|
||||
export function parseTaskLog(
|
||||
logStr?: string | undefined,
|
||||
): typeof TaskLog.infer {
|
||||
if (!logStr) return { message: "", timestamp: "", level: "" };
|
||||
const log = JSON.parse(logStr);
|
||||
|
||||
if (typeof log.level === "number") {
|
||||
log.level = labelNumberMap[
|
||||
log.level as keyof typeof labelNumberMap
|
||||
] as string;
|
||||
}
|
||||
|
||||
return {
|
||||
message: log.msg,
|
||||
timestamp: log.time,
|
||||
level: log.level,
|
||||
};
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user