Compare commits
109 Commits
Author | SHA1 | Date | |
---|---|---|---|
35ba25ba9e | |||
724dca0e69 | |||
9029f72a02 | |||
1d55d7f2d2 | |||
d09bb4e0c6 | |||
53c7c9d9e2 | |||
df9c60c29e | |||
f29b6b23f0 | |||
cc88fa6763 | |||
19022e7cb5 | |||
44f43dbc2c | |||
45f6f3af32 | |||
9bd544f0d5 | |||
a315e58284 | |||
d576b71bc5 | |||
e02f7f59c6 | |||
e8c6467faa | |||
43f67c795d | |||
83f0902291 | |||
11a6621bd7 | |||
36fa451249 | |||
d112eba8fe | |||
ef39f9ec0b | |||
8517c20e6c | |||
728ad56b33 | |||
f185ccee03 | |||
284bbcbae2 | |||
7eba454f1b | |||
dcd9efaeba | |||
f13bcbaf3f | |||
8e7e0414a6 | |||
229c8d8368 | |||
1adf87b884 | |||
a061b540b1 | |||
824603ff4a | |||
539d99d35f | |||
716034660c | |||
bab65bcdad | |||
a0732a4593 | |||
835f7df0f6 | |||
2def365cae | |||
c8b2539414 | |||
c8a25eb9bf | |||
15bceb1715 | |||
8acc4108ae | |||
0f708f53c0 | |||
8d03ad8e15 | |||
33829768eb | |||
8cee6579e2 | |||
8eeb2a8de7 | |||
e4ce767444 | |||
00fd514da5 | |||
ba38d2b819 | |||
d0f83e6148 | |||
a040ad469e | |||
0c01866344 | |||
048c5198a2 | |||
f606916d97 | |||
aea85c3b73 | |||
5f904b3971 | |||
ac5d52a010 | |||
8644a937d0 | |||
a0ebb97bc5 | |||
19cd9b8507 | |||
fe653e91c8 | |||
a78ec72687 | |||
a462bed948 | |||
325518ba15 | |||
4b80912727 | |||
|
c78ad636c3 | ||
4fe7833977 | |||
24f9aeb229 | |||
976413e11b | |||
2480d875b4 | |||
5dff12ea70 | |||
e332dc63e8 | |||
a18e9bcc88 | |||
4b43edf54a | |||
e3aa2dd5ff | |||
98a03c25e6 | |||
7f618e7e45 | |||
84a7be4549 | |||
b2bad5f64c | |||
41f55c0920 | |||
edbf8a8ecf | |||
4852804581 | |||
22b1c35b3e | |||
d1e9ffd89b | |||
9c675ba48c | |||
ef3d3d5bde | |||
7e835af3f2 | |||
f153102978 | |||
5944c784ba | |||
2aa186450a | |||
70b53106ea | |||
b958189b0c | |||
acd488aab0 | |||
5378b7a820 | |||
1bb5d3ecb0 | |||
0d79e9c85e | |||
70382a63ed | |||
b3886cc6fc | |||
cfe525f274 | |||
96e8ec4b67 | |||
7a3397f978 | |||
dc212d7441 | |||
6e04594b4d | |||
b6834aa829 | |||
093a6c5c2b |
@@ -1 +1,15 @@
|
|||||||
target
|
*
|
||||||
|
!/entry.sh
|
||||||
|
|
||||||
|
!/backend/src
|
||||||
|
!/backend/Cargo.lock
|
||||||
|
!/backend/Cargo.toml
|
||||||
|
|
||||||
|
!/frontend/locales
|
||||||
|
!/frontend/src
|
||||||
|
!/frontend/static
|
||||||
|
!/frontend/.npmrc
|
||||||
|
!/frontend/package.json
|
||||||
|
!/frontend/pnpm-lock.yaml
|
||||||
|
!/frontend/svelte.config.js
|
||||||
|
!/frontend/tsconfig.json
|
||||||
|
BIN
.github/lokalise.png
vendored
Normal file
After Width: | Height: | Size: 30 KiB |
3
.github/workflows/docker.yml
vendored
@@ -14,6 +14,8 @@ jobs:
|
|||||||
uses: docker/setup-qemu-action@v1
|
uses: docker/setup-qemu-action@v1
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v1
|
uses: docker/setup-buildx-action@v1
|
||||||
|
with:
|
||||||
|
install: true
|
||||||
- name: Docker Labels
|
- name: Docker Labels
|
||||||
id: meta
|
id: meta
|
||||||
uses: crazy-max/ghaction-docker-meta@v2
|
uses: crazy-max/ghaction-docker-meta@v2
|
||||||
@@ -32,6 +34,7 @@ jobs:
|
|||||||
id: docker_build
|
id: docker_build
|
||||||
uses: docker/build-push-action@v2
|
uses: docker/build-push-action@v2
|
||||||
with:
|
with:
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
push: true
|
push: true
|
||||||
tags: ${{ steps.meta.outputs.tags }}
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
- name: Image digest
|
- name: Image digest
|
||||||
|
3
.gitignore
vendored
@@ -1,6 +1,6 @@
|
|||||||
|
|
||||||
# Backend
|
# Backend
|
||||||
/target
|
target
|
||||||
|
|
||||||
# Client
|
# Client
|
||||||
.DS_Store
|
.DS_Store
|
||||||
@@ -8,3 +8,4 @@ node_modules
|
|||||||
/.svelte
|
/.svelte
|
||||||
/build
|
/build
|
||||||
/functions
|
/functions
|
||||||
|
.env
|
||||||
|
10
.vscode/settings.json
vendored
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"cSpell.words": [
|
"cSpell.words": ["ciphertext", "cryptgeon"],
|
||||||
"ciphertext",
|
"i18n-ally.localesPaths": ["frontend/locales"],
|
||||||
"cryptgeon"
|
"i18n-ally.enabledFrameworks": ["svelte"],
|
||||||
]
|
"i18n-ally.keystyle": "nested"
|
||||||
}
|
}
|
||||||
|
172
CHANGELOG.md
@@ -5,35 +5,193 @@ All notable changes to this project will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [1.5.2] - 2022-06-07
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Wait for script for memecached.
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
- Updated dependencies.
|
||||||
|
|
||||||
|
## [1.5.1] - 2022-05-15
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Remove double note content.
|
||||||
|
|
||||||
|
## [1.5.0] - 2022-05-14
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Links in notes are not highlighted and can be directly clicked #30.
|
||||||
|
|
||||||
|
## [1.4.1] - 2022-03-05
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Router in prod build.
|
||||||
|
|
||||||
|
## [1.4.0] - 2022-03-02
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Support for multiple languages.
|
||||||
|
- Select multiple files without removing already selected ones.
|
||||||
|
- Tooltip for copy action.
|
||||||
|
- Configure maximum views, expiration and advanced options for the server.
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Use native SVGs instead of images.
|
||||||
|
- Update robots.txt file to allow only root.
|
||||||
|
- Stronger frontend types.
|
||||||
|
|
||||||
|
## [1.3.3] - 2022-01-03
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Bug fix due to dependency update.
|
||||||
|
|
||||||
|
## [1.3.2] - 2022-01-02
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Dependencies updates.
|
||||||
|
- Folder structure.
|
||||||
|
|
||||||
|
## [1.3.1] - 2021-12-30
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Short explanation in the home page.
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Explanation in about & readme.
|
||||||
|
- Shorten server ids from 512 to 256bit.
|
||||||
|
|
||||||
|
## [1.3.0] - 2021-12-22
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Option to set a custom size limit.
|
||||||
|
- Options to share files.
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Don't delete note if time is not expired yet
|
||||||
|
- Use pnpm instead of npm.
|
||||||
|
|
||||||
|
## [1.2.0] - 2021-11-11
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Switch to pnpm.
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
- Dependencies updated.
|
||||||
|
|
||||||
|
## [1.1.1] - 2021-05-17
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Height on big displays.
|
||||||
|
- About page.
|
||||||
|
|
||||||
|
## [1.1.0] - 2021-05-16
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
- Using hash `#` instead of path.
|
||||||
|
|
||||||
|
## [1.0.11] - 2021-05-08
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- loading text.
|
||||||
|
- description for created notes about availability.
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- iterations from 100 to 100k.
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- time based view bug.
|
||||||
|
|
||||||
|
## [1.0.10] - 2021-05-08
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- API endpoint was not reachable.
|
||||||
|
|
||||||
|
## [1.0.9] - 2021-05-07
|
||||||
|
|
||||||
|
## Changed
|
||||||
|
|
||||||
|
- Removed a dependency.
|
||||||
|
|
||||||
|
## [1.0.8] - 2021-05-05
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Manual theme override option.
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Removed Arm builds for now.
|
||||||
|
- iOS style bugs.
|
||||||
|
|
||||||
|
## [1.0.7] - 2021-05-04
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Arm images.
|
||||||
|
|
||||||
|
## [1.0.6] - 2021-05-04
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Always use encryption with random passwords included links.
|
||||||
|
|
||||||
|
## [1.0.5] - 2021-05-03
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Typos.
|
||||||
|
|
||||||
## [1.0.4] - 2021-05-02
|
## [1.0.4] - 2021-05-02
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- From scratch docker image
|
- From scratch docker image.
|
||||||
|
|
||||||
## [1.0.3] - 2021-05-02
|
## [1.0.3] - 2021-05-02
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Higher default text area
|
- Higher default text area.
|
||||||
- Mobile touchups
|
- Mobile touchups.
|
||||||
|
|
||||||
## [1.0.2] - 2021-05-02
|
## [1.0.2] - 2021-05-02
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- SVG Icons
|
- SVG Icons.
|
||||||
|
|
||||||
## [1.0.1] - 2021-05-02
|
## [1.0.1] - 2021-05-02
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Dark mode support
|
- Dark mode support.
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Don't reload data on wrong password
|
- Don't reload data on wrong password.
|
||||||
|
|
||||||
## [1.0.0] - 2021-05-02
|
## [1.0.0] - 2021-05-02
|
||||||
|
|
||||||
Initial release
|
Initial release.
|
||||||
|
2030
Cargo.lock
generated
32
Dockerfile
@@ -1,26 +1,26 @@
|
|||||||
FROM node:16-alpine as CLIENT
|
# FRONTEND
|
||||||
|
FROM node:16-alpine as client
|
||||||
WORKDIR /tmp
|
WORKDIR /tmp
|
||||||
COPY ./client .
|
RUN npm install -g pnpm
|
||||||
|
COPY ./frontend ./
|
||||||
|
RUN pnpm install
|
||||||
|
RUN pnpm run build
|
||||||
|
|
||||||
RUN npm ci
|
|
||||||
RUN npm run build
|
|
||||||
|
|
||||||
FROM rust:1.51-alpine as RUST
|
|
||||||
|
|
||||||
|
# BACKEND
|
||||||
|
FROM rust:1.61-alpine as backend
|
||||||
WORKDIR /tmp
|
WORKDIR /tmp
|
||||||
RUN apk add libc-dev openssl-dev alpine-sdk
|
RUN apk add libc-dev openssl-dev alpine-sdk
|
||||||
COPY ./Cargo* .
|
COPY ./backend ./
|
||||||
COPY ./src ./src
|
|
||||||
|
|
||||||
RUN cargo build --release
|
RUN cargo build --release
|
||||||
|
|
||||||
FROM scratch
|
|
||||||
|
|
||||||
|
# RUNNER
|
||||||
|
FROM alpine
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --from=RUST /tmp/target/release/cryptgeon .
|
COPY ./entry.sh .
|
||||||
COPY --from=CLIENT /tmp/build ./client/build
|
COPY --from=backend /tmp/target/release/cryptgeon .
|
||||||
|
COPY --from=client /tmp/build ./frontend/build
|
||||||
ENV MEMCACHE=memcached:11211
|
ENV MEMCACHE=memcached:11211
|
||||||
|
EXPOSE 5000
|
||||||
ENTRYPOINT [ "/app/cryptgeon" ]
|
ENTRYPOINT [ "/app/entry.sh" ]
|
||||||
|
128
README.md
@@ -1,14 +1,26 @@
|
|||||||
<p align="center">
|
<p align="center">
|
||||||
<img src="./design/Github.png">
|
<img src="./design/Github.png" alt="logo">
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||

|
<a href="https://discord.gg/nuby6RnxZt">
|
||||||

|
<img alt="discord" src="https://img.shields.io/discord/252403122348097536?style=for-the-badge" />
|
||||||

|
<img alt="docker pulls" src="https://img.shields.io/docker/pulls/cupcakearmy/cryptgeon?style=for-the-badge" />
|
||||||
|
<img alt="Docker image size badge" src="https://img.shields.io/docker/image-size/cupcakearmy/cryptgeon?style=for-the-badge" />
|
||||||
|
<img alt="Latest version" src="https://img.shields.io/github/v/release/cupcakearmy/cryptgeon?style=for-the-badge" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
<a href="https://www.producthunt.com/posts/cryptgeon?utm_source=badge-featured&utm_medium=badge&utm_souce=badge-cryptgeon" target="_blank"><img src="https://api.producthunt.com/widgets/embed-image/v1/featured.svg?post_id=295189&theme=light" alt="Cryptgeon - Securely share self-destructing notes | Product Hunt" height="50" /></a>
|
||||||
|
<a href=""><img src="./.github/lokalise.png" height="50">
|
||||||
|
<br/>
|
||||||
|
|
||||||
## About?
|
## About?
|
||||||
|
|
||||||
_cryptgeon_ is an secure, open source sharing note service inspired by [_PrivNote_](https://privnote.com)
|
_cryptgeon_ is a secure, open source sharing note or file service inspired by [_PrivNote_](https://privnote.com)
|
||||||
|
|
||||||
|
> 🌍 If you want to translate the project feel free to reach out to me.
|
||||||
|
>
|
||||||
|
> Thanks to [Lokalise](https://lokalise.com/) for providing free access to their platform.
|
||||||
|
|
||||||
## Demo
|
## Demo
|
||||||
|
|
||||||
@@ -16,21 +28,40 @@ Check out the demo and see for yourself https://cryptgeon.nicco.io.
|
|||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- view and time constrains
|
- server cannot decrypt contents due to client side encryption
|
||||||
|
- view or time constraints
|
||||||
- in memory, no persistence
|
- in memory, no persistence
|
||||||
- in browser encryption → server cannot decrypt contents
|
|
||||||
- obligatory dark mode support
|
- obligatory dark mode support
|
||||||
|
|
||||||
## How does it work?
|
## How does it work?
|
||||||
|
|
||||||
each note has a 512bit generated <i>id</i> that is used to retrieve the note. data is stored in memory and never persisted to disk.
|
each note has a generated <code>id (256bit)</code> and <code>key 256(bit)</code>. The
|
||||||
|
<code>id</code>
|
||||||
|
is used to save & retrieve the note. the note is then encrypted with aes in gcm mode on the
|
||||||
|
client side with the <code>key</code> and then sent to the server. data is stored in memory and
|
||||||
|
never persisted to disk. the server never sees the encryption key and cannot decrypt the contents
|
||||||
|
of the notes even if it tried to.
|
||||||
|
|
||||||
## Screenshot
|
## Screenshot
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
|
## Environment Variables
|
||||||
|
|
||||||
|
| Variable | Default | Description |
|
||||||
|
| ---------------- | ----------------- | --------------------------------------------------------------------------------------- |
|
||||||
|
| `MEMCACHE` | `memcached:11211` | Memcached URL to connect to. |
|
||||||
|
| `SIZE_LIMIT` | `1 KiB` | Max size for body. Accepted values according to [byte-unit](https://docs.rs/byte-unit/) |
|
||||||
|
| `MAX_VIEWS` | `100` | Maximal number of views. |
|
||||||
|
| `MAX_EXPIRATION` | `360` | Maximal expiration in minutes. |
|
||||||
|
| `ALLOW_ADVANCED` | `true` | Allow custom configuration. If set to `false` all notes will be one view only. |
|
||||||
|
|
||||||
## Deployment
|
## Deployment
|
||||||
|
|
||||||
|
ℹ️ `https` is required otherwise browsers will not support the cryptographic functions.
|
||||||
|
|
||||||
|
### Docker
|
||||||
|
|
||||||
Docker is the easiest way. There is the [official image here](https://hub.docker.com/r/cupcakearmy/cryptgeon).
|
Docker is the easiest way. There is the [official image here](https://hub.docker.com/r/cupcakearmy/cryptgeon).
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
@@ -41,27 +72,96 @@ version: '3.7'
|
|||||||
services:
|
services:
|
||||||
memcached:
|
memcached:
|
||||||
image: memcached:1-alpine
|
image: memcached:1-alpine
|
||||||
entrypoint: memcached -m 128 # Limit to 128 MB Ram, customize at free will.
|
entrypoint: memcached -m 128M -I 4M # Limit to 128 MB Ram, 4M per entry, customize at free will.
|
||||||
|
|
||||||
app:
|
app:
|
||||||
image: cupcakearmy/cryptgeon:latest
|
image: cupcakearmy/cryptgeon:latest
|
||||||
|
depends_on:
|
||||||
|
- memcached
|
||||||
|
environment:
|
||||||
|
SIZE_LIMIT: 4M
|
||||||
ports:
|
ports:
|
||||||
- 80:5000
|
- 80:5000
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### NGINX Proxy
|
||||||
|
|
||||||
|
See the [examples/nginx](https://github.com/cupcakearmy/cryptgeon/tree/main/examples/nginx) folder. There an example with a simple proxy, and one with https. You need to specify the server names and certificates.
|
||||||
|
|
||||||
|
### Traefik 2
|
||||||
|
|
||||||
|
Assumptions:
|
||||||
|
|
||||||
|
- External proxy docker network `proxy`
|
||||||
|
- A certificate resolver `le`
|
||||||
|
- A https entrypoint `secure`
|
||||||
|
- Domain name `example.org`
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
networks:
|
||||||
|
proxy:
|
||||||
|
external: true
|
||||||
|
|
||||||
|
services:
|
||||||
|
memcached:
|
||||||
|
image: memcached:1-alpine
|
||||||
|
restart: unless-stopped
|
||||||
|
entrypoint: memcached -m 128M -I 4M # Limit to 128 MB Ram, 4M per entry, customize at free will.
|
||||||
|
|
||||||
|
app:
|
||||||
|
image: cupcakearmy/cryptgeon:latest
|
||||||
|
restart: unless-stopped
|
||||||
|
depends_on:
|
||||||
|
- memcached
|
||||||
|
networks:
|
||||||
|
- default
|
||||||
|
- proxy
|
||||||
|
labels:
|
||||||
|
- traefik.enable=true
|
||||||
|
- traefik.http.routers.cryptgeon.rule=Host(`example.org`)
|
||||||
|
- traefik.http.routers.cryptgeon.entrypoints=secure
|
||||||
|
- traefik.http.routers.cryptgeon.tls.certresolver=le
|
||||||
|
```
|
||||||
|
|
||||||
## Development
|
## Development
|
||||||
|
|
||||||
1. Clone
|
**Requirements**
|
||||||
2. run `npm i` in the root and and client `client/` folders.
|
|
||||||
3. Run `npm run dev` to start development.
|
|
||||||
|
|
||||||
Running `npm run dev` in the root folder will start the following things
|
- `pnpm`: `>=6`
|
||||||
|
- `node`: `>=14`
|
||||||
|
- `rust`: edition `2021`
|
||||||
|
|
||||||
|
**Install**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm install
|
||||||
|
pnpm --prefix frontend install
|
||||||
|
|
||||||
|
# Also you need cargo watch if you don't already have it installed.
|
||||||
|
# https://lib.rs/crates/cargo-watch
|
||||||
|
cargo install cargo-watch
|
||||||
|
```
|
||||||
|
|
||||||
|
**Run**
|
||||||
|
|
||||||
|
Make sure you have docker running.
|
||||||
|
|
||||||
|
> If you are on `macOS` you might need to disable AirPlay Receiver as it uses port 5000 (So stupid...)
|
||||||
|
> https://developer.apple.com/forums/thread/682332
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm run dev
|
||||||
|
```
|
||||||
|
|
||||||
|
Running `pnpm run dev` in the root folder will start the following things:
|
||||||
|
|
||||||
- a memcache docker container
|
- a memcache docker container
|
||||||
- rust backend with hot reload
|
- rust backend with hot reload
|
||||||
- client with hot reload
|
- client with hot reload
|
||||||
|
|
||||||
You can see the app under [localhost:3000](http://localhost:3000).
|
You can see the app under [localhost:1234](http://localhost:1234).
|
||||||
|
|
||||||
###### Attributions
|
###### Attributions
|
||||||
|
|
||||||
|
1551
backend/Cargo.lock
generated
Normal file
@@ -1,8 +1,8 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "cryptgeon"
|
name = "cryptgeon"
|
||||||
version = "1.0.0"
|
version = "1.5.2"
|
||||||
authors = ["cupcakearmy <hi@nicco.io>"]
|
authors = ["cupcakearmy <hi@nicco.io>"]
|
||||||
edition = "2018"
|
edition = "2021"
|
||||||
|
|
||||||
[[bin]]
|
[[bin]]
|
||||||
name = "cryptgeon"
|
name = "cryptgeon"
|
||||||
@@ -11,11 +11,14 @@ path = "src/main.rs"
|
|||||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
actix-web = "3"
|
actix-web = "4"
|
||||||
actix-files = "0.5"
|
actix-files = "0.6"
|
||||||
serde = "1"
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
serde_json = "1"
|
serde_json = "1"
|
||||||
lazy_static = "1"
|
lazy_static = "1"
|
||||||
ring = "0.16"
|
ring = "0.16"
|
||||||
bs62 = "0.1"
|
bs62 = "0.1"
|
||||||
memcache = "0.15"
|
memcache = "0.16"
|
||||||
|
byte-unit = "4"
|
||||||
|
dotenv = "0.15"
|
||||||
|
mime = "0.3"
|
12
backend/src/api.rs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
use actix_web::web;
|
||||||
|
|
||||||
|
use crate::note;
|
||||||
|
use crate::status;
|
||||||
|
|
||||||
|
pub fn init(cfg: &mut web::ServiceConfig) {
|
||||||
|
cfg.service(
|
||||||
|
web::scope("/api")
|
||||||
|
.service(note::init())
|
||||||
|
.service(status::init()),
|
||||||
|
);
|
||||||
|
}
|
14
backend/src/client.rs
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
use actix_files::{Files, NamedFile};
|
||||||
|
use actix_web::{web, Result};
|
||||||
|
|
||||||
|
pub fn init(cfg: &mut web::ServiceConfig) {
|
||||||
|
cfg.service(
|
||||||
|
Files::new("/", "./frontend/build")
|
||||||
|
.index_file("index.html")
|
||||||
|
.use_etag(true),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn index() -> Result<NamedFile> {
|
||||||
|
Ok(NamedFile::open("./frontend/build/index.html")?)
|
||||||
|
}
|
23
backend/src/config.rs
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
use byte_unit::Byte;
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
pub static ref VERSION: String = option_env!("CARGO_PKG_VERSION")
|
||||||
|
.unwrap_or("Unknown")
|
||||||
|
.to_string();
|
||||||
|
pub static ref LIMIT: u32 =
|
||||||
|
Byte::from_str(std::env::var("SIZE_LIMIT").unwrap_or("1 KiB".to_string()))
|
||||||
|
.unwrap()
|
||||||
|
.get_bytes() as u32;
|
||||||
|
pub static ref MAX_VIEWS: u32 = std::env::var("MAX_VIEWS")
|
||||||
|
.unwrap_or("100".to_string())
|
||||||
|
.parse()
|
||||||
|
.unwrap();
|
||||||
|
pub static ref MAX_EXPIRATION: u32 = std::env::var("MAX_EXPIRATION")
|
||||||
|
.unwrap_or("360".to_string()) // 6 hours in minutes
|
||||||
|
.parse()
|
||||||
|
.unwrap();
|
||||||
|
pub static ref ALLOW_ADVANCED: bool = std::env::var("ALLOW_ADVANCED")
|
||||||
|
.unwrap_or("true".to_string())
|
||||||
|
.parse()
|
||||||
|
.unwrap();
|
||||||
|
}
|
@@ -1,23 +1,30 @@
|
|||||||
use actix_web::{middleware, web, App, HttpServer};
|
use actix_web::{middleware, web, App, HttpServer};
|
||||||
|
use dotenv::dotenv;
|
||||||
|
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate lazy_static;
|
extern crate lazy_static;
|
||||||
|
|
||||||
|
mod api;
|
||||||
mod client;
|
mod client;
|
||||||
|
mod config;
|
||||||
mod note;
|
mod note;
|
||||||
|
mod size;
|
||||||
|
mod status;
|
||||||
mod store;
|
mod store;
|
||||||
|
|
||||||
#[actix_web::main]
|
#[actix_web::main]
|
||||||
async fn main() -> std::io::Result<()> {
|
async fn main() -> std::io::Result<()> {
|
||||||
HttpServer::new(|| {
|
dotenv().ok();
|
||||||
|
return HttpServer::new(|| {
|
||||||
App::new()
|
App::new()
|
||||||
.wrap(middleware::Compress::default())
|
.wrap(middleware::Compress::default())
|
||||||
.wrap(middleware::DefaultHeaders::default())
|
.wrap(middleware::DefaultHeaders::default())
|
||||||
.configure(note::init)
|
.configure(size::init)
|
||||||
|
.configure(api::init)
|
||||||
.configure(client::init)
|
.configure(client::init)
|
||||||
.default_service(web::resource("").route(web::get().to(client::fallback_fn)))
|
.default_service(web::to(client::index))
|
||||||
})
|
})
|
||||||
.bind("0.0.0.0:5000")?
|
.bind("0.0.0.0:5000")?
|
||||||
.run()
|
.run()
|
||||||
.await
|
.await;
|
||||||
}
|
}
|
27
backend/src/note/model.rs
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
use bs62;
|
||||||
|
use ring::rand::SecureRandom;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize, Clone)]
|
||||||
|
pub struct Note {
|
||||||
|
pub meta: String,
|
||||||
|
pub contents: String,
|
||||||
|
pub views: Option<u32>,
|
||||||
|
pub expiration: Option<u32>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize, Clone)]
|
||||||
|
pub struct NoteInfo {}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize, Clone)]
|
||||||
|
pub struct NotePublic {
|
||||||
|
pub meta: String,
|
||||||
|
pub contents: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn generate_id() -> String {
|
||||||
|
let mut id: [u8; 32] = [0; 32];
|
||||||
|
let sr = ring::rand::SystemRandom::new();
|
||||||
|
let _ = sr.fill(&mut id);
|
||||||
|
return bs62::encode_data(&id);
|
||||||
|
}
|
124
backend/src/note/routes.rs
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
use actix_web::{delete, get, post, web, HttpResponse, Responder, Scope};
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
use std::time::SystemTime;
|
||||||
|
|
||||||
|
use crate::config;
|
||||||
|
use crate::note::{generate_id, Note, NoteInfo, NotePublic};
|
||||||
|
use crate::store;
|
||||||
|
|
||||||
|
pub fn now() -> u32 {
|
||||||
|
SystemTime::now()
|
||||||
|
.duration_since(SystemTime::UNIX_EPOCH)
|
||||||
|
.unwrap()
|
||||||
|
.as_secs() as u32
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
struct NotePath {
|
||||||
|
id: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[get("/{id}")]
|
||||||
|
async fn one(path: web::Path<NotePath>) -> impl Responder {
|
||||||
|
let p = path.into_inner();
|
||||||
|
let note = store::get(&p.id);
|
||||||
|
match note {
|
||||||
|
None => return HttpResponse::NotFound().finish(),
|
||||||
|
Some(_) => return HttpResponse::Ok().json(NoteInfo {}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
struct CreateResponse {
|
||||||
|
id: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[post("/")]
|
||||||
|
async fn create(note: web::Json<Note>) -> impl Responder {
|
||||||
|
let mut n = note.into_inner();
|
||||||
|
let id = generate_id();
|
||||||
|
let bad_req = HttpResponse::BadRequest().finish();
|
||||||
|
if n.views == None && n.expiration == None {
|
||||||
|
return bad_req;
|
||||||
|
}
|
||||||
|
if !*config::ALLOW_ADVANCED {
|
||||||
|
n.views = Some(1);
|
||||||
|
n.expiration = None;
|
||||||
|
}
|
||||||
|
match n.views {
|
||||||
|
Some(v) => {
|
||||||
|
if v > *config::MAX_VIEWS {
|
||||||
|
return bad_req;
|
||||||
|
}
|
||||||
|
n.expiration = None; // views overrides expiration
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
match n.expiration {
|
||||||
|
Some(e) => {
|
||||||
|
if e > *config::MAX_EXPIRATION {
|
||||||
|
return bad_req;
|
||||||
|
}
|
||||||
|
let expiration = now() + (e * 60);
|
||||||
|
n.expiration = Some(expiration);
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
store::set(&id.clone(), &n.clone());
|
||||||
|
return HttpResponse::Ok().json(CreateResponse { id: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
#[delete("/{id}")]
|
||||||
|
async fn delete(path: web::Path<NotePath>) -> impl Responder {
|
||||||
|
let p = path.into_inner();
|
||||||
|
let note = store::get(&p.id);
|
||||||
|
match note {
|
||||||
|
None => return HttpResponse::NotFound().finish(),
|
||||||
|
Some(note) => {
|
||||||
|
let mut changed = note.clone();
|
||||||
|
if changed.views == None && changed.expiration == None {
|
||||||
|
return HttpResponse::BadRequest().finish();
|
||||||
|
}
|
||||||
|
match changed.views {
|
||||||
|
Some(v) => {
|
||||||
|
changed.views = Some(v - 1);
|
||||||
|
let id = p.id.clone();
|
||||||
|
if v <= 1 {
|
||||||
|
store::del(&id);
|
||||||
|
} else {
|
||||||
|
store::set(&id, &changed.clone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
|
||||||
|
let n = now();
|
||||||
|
match changed.expiration {
|
||||||
|
Some(e) => {
|
||||||
|
if e < n {
|
||||||
|
store::del(&p.id.clone());
|
||||||
|
return HttpResponse::BadRequest().finish();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
return HttpResponse::Ok().json(NotePublic {
|
||||||
|
contents: changed.contents,
|
||||||
|
meta: changed.meta,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
struct Status {
|
||||||
|
version: String,
|
||||||
|
max_size: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn init() -> Scope {
|
||||||
|
web::scope("/notes")
|
||||||
|
.service(one)
|
||||||
|
.service(create)
|
||||||
|
.service(delete)
|
||||||
|
}
|
18
backend/src/size.rs
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
use actix_web::web;
|
||||||
|
use byte_unit::Byte;
|
||||||
|
use mime;
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
pub static ref LIMIT: usize =
|
||||||
|
Byte::from_str(std::env::var("SIZE_LIMIT").unwrap_or("1 KiB".to_string()))
|
||||||
|
.unwrap()
|
||||||
|
.get_bytes() as usize;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn init(cfg: &mut web::ServiceConfig) {
|
||||||
|
let json = web::JsonConfig::default().limit(*LIMIT);
|
||||||
|
let plain = web::PayloadConfig::default()
|
||||||
|
.limit(*LIMIT)
|
||||||
|
.mimetype(mime::STAR_STAR);
|
||||||
|
cfg.app_data(json).app_data(plain);
|
||||||
|
}
|
5
backend/src/status/mod.rs
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
mod model;
|
||||||
|
mod routes;
|
||||||
|
|
||||||
|
pub use model::*;
|
||||||
|
pub use routes::*;
|
10
backend/src/status/model.rs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
pub struct Status {
|
||||||
|
pub version: String,
|
||||||
|
pub max_size: u32,
|
||||||
|
pub max_views: u32,
|
||||||
|
pub max_expiration: u32,
|
||||||
|
pub allow_advanced: bool,
|
||||||
|
}
|
19
backend/src/status/routes.rs
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
use actix_web::{get, web, HttpResponse, Responder, Scope};
|
||||||
|
|
||||||
|
use crate::config;
|
||||||
|
use crate::status::Status;
|
||||||
|
|
||||||
|
#[get("/")]
|
||||||
|
async fn get_status() -> impl Responder {
|
||||||
|
return HttpResponse::Ok().json(Status {
|
||||||
|
version: config::VERSION.to_string(),
|
||||||
|
max_size: *config::LIMIT,
|
||||||
|
max_views: *config::MAX_VIEWS,
|
||||||
|
max_expiration: *config::MAX_EXPIRATION,
|
||||||
|
allow_advanced: *config::ALLOW_ADVANCED,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn init() -> Scope {
|
||||||
|
web::scope("/status").service(get_status)
|
||||||
|
}
|
36
backend/src/store.rs
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
use memcache;
|
||||||
|
|
||||||
|
use crate::note::now;
|
||||||
|
use crate::note::Note;
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
static ref CLIENT: memcache::Client = memcache::connect(format!(
|
||||||
|
"memcache://{}?timeout=10&tcp_nodelay=true",
|
||||||
|
std::env::var("MEMCACHE").unwrap_or("127.0.0.1:11211".to_string())
|
||||||
|
))
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set(id: &String, note: &Note) {
|
||||||
|
let serialized = serde_json::to_string(¬e.clone()).unwrap();
|
||||||
|
let expiration: u32 = match note.expiration {
|
||||||
|
Some(e) => e - now(),
|
||||||
|
None => 0,
|
||||||
|
};
|
||||||
|
CLIENT.set(id, serialized, expiration).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get(id: &String) -> Option<Note> {
|
||||||
|
let value: Option<String> = CLIENT.get(&id).unwrap();
|
||||||
|
match value {
|
||||||
|
None => return None,
|
||||||
|
Some(s) => {
|
||||||
|
let deserialize: Note = serde_json::from_str(&s).unwrap();
|
||||||
|
return Some(deserialize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn del(id: &String) {
|
||||||
|
CLIENT.delete(id).unwrap();
|
||||||
|
}
|
@@ -1,38 +0,0 @@
|
|||||||
# create-svelte
|
|
||||||
|
|
||||||
Everything you need to build a Svelte project, powered by [`create-svelte`](https://github.com/sveltejs/kit/tree/master/packages/create-svelte);
|
|
||||||
|
|
||||||
## Creating a project
|
|
||||||
|
|
||||||
If you're seeing this, you've probably already done this step. Congrats!
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# create a new project in the current directory
|
|
||||||
npm init svelte@next
|
|
||||||
|
|
||||||
# create a new project in my-app
|
|
||||||
npm init svelte@next my-app
|
|
||||||
```
|
|
||||||
|
|
||||||
> Note: the `@next` is temporary
|
|
||||||
|
|
||||||
## Developing
|
|
||||||
|
|
||||||
Once you've created a project and installed dependencies with `npm install` (or `pnpm install` or `yarn`), start a development server:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run dev
|
|
||||||
|
|
||||||
# or start the server and open the app in a new browser tab
|
|
||||||
npm run dev -- --open
|
|
||||||
```
|
|
||||||
|
|
||||||
## Building
|
|
||||||
|
|
||||||
Before creating a production version of your app, install an [adapter](https://kit.svelte.dev/docs#adapters) for your target environment. Then:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run build
|
|
||||||
```
|
|
||||||
|
|
||||||
> You can preview the built app with `npm run preview`, regardless of whether you installed an adapter. This should _not_ be used to serve your app in production.
|
|
1000
client/package-lock.json
generated
@@ -1,24 +0,0 @@
|
|||||||
{
|
|
||||||
"private": true,
|
|
||||||
"scripts": {
|
|
||||||
"dev": "svelte-kit dev",
|
|
||||||
"build": "svelte-kit build",
|
|
||||||
"preview": "svelte-kit preview",
|
|
||||||
"licenses": "npx license-checker --summary > licenses.csv"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@sveltejs/adapter-static": "next",
|
|
||||||
"@sveltejs/kit": "next",
|
|
||||||
"svelte": "^3.34.0",
|
|
||||||
"svelte-preprocess": "^4.0.0",
|
|
||||||
"tslib": "^2.0.0",
|
|
||||||
"typescript": "^4.0.0",
|
|
||||||
"vite": "^2.1.0"
|
|
||||||
},
|
|
||||||
"type": "module",
|
|
||||||
"dependencies": {
|
|
||||||
"@fontsource/fira-mono": "^4.2.2",
|
|
||||||
"axios": "^0.21.1",
|
|
||||||
"copy-to-clipboard": "^3.3.1"
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,80 +0,0 @@
|
|||||||
@import '@fontsource/fira-mono';
|
|
||||||
|
|
||||||
* {
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
|
||||||
|
|
||||||
:root {
|
|
||||||
font-family: 'Fira Mono', monospace;
|
|
||||||
|
|
||||||
--ui-bg-0: #fefefe;
|
|
||||||
--ui-bg-0-85: #fefefed9;
|
|
||||||
--ui-bg-1: #eee;
|
|
||||||
--ui-bg-2: #e2e2e2;
|
|
||||||
--ui-text-0: #111;
|
|
||||||
--ui-text-1: #222;
|
|
||||||
--ui-clr-primary: hsl(186, 65%, 55%);
|
|
||||||
--ui-clr-error: hsl(357, 77%, 51%);
|
|
||||||
|
|
||||||
--ui-anim: all 150ms ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (prefers-color-scheme: dark) {
|
|
||||||
:root {
|
|
||||||
--ui-bg-0: #111;
|
|
||||||
--ui-bg-0-85: #111111d9;
|
|
||||||
--ui-bg-1: #222;
|
|
||||||
--ui-bg-2: #282828;
|
|
||||||
--ui-text-0: #fefefe;
|
|
||||||
--ui-text-1: #eee;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.error-text {
|
|
||||||
color: var(--ui-clr-error);
|
|
||||||
}
|
|
||||||
|
|
||||||
body {
|
|
||||||
min-height: 100vh;
|
|
||||||
margin: 0;
|
|
||||||
background-color: var(--ui-bg-0);
|
|
||||||
color: var(--ui-text-0);
|
|
||||||
}
|
|
||||||
|
|
||||||
a {
|
|
||||||
color: inherit;
|
|
||||||
text-decoration: inherit;
|
|
||||||
box-sizing: content-box;
|
|
||||||
border-bottom: 2px solid var(--ui-bg-2);
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
|
|
||||||
a:active {
|
|
||||||
border-color: var(--ui-clr-primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
a:hover {
|
|
||||||
border-color: var(--ui-text-0);
|
|
||||||
}
|
|
||||||
|
|
||||||
input,
|
|
||||||
textarea,
|
|
||||||
button {
|
|
||||||
appearance: none;
|
|
||||||
transition: var(--ui-anim);
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: inherit;
|
|
||||||
background: inherit;
|
|
||||||
color: inherit;
|
|
||||||
}
|
|
||||||
|
|
||||||
*:disabled,
|
|
||||||
*[disabled='true'] {
|
|
||||||
opacity: 0.5;
|
|
||||||
}
|
|
||||||
|
|
||||||
fieldset {
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
border: none;
|
|
||||||
}
|
|
@@ -1,38 +0,0 @@
|
|||||||
import axios from 'axios'
|
|
||||||
import { dev } from '$app/env'
|
|
||||||
|
|
||||||
const base = axios.create({ baseURL: dev ? 'http://localhost:5000' : undefined })
|
|
||||||
|
|
||||||
export type Note = {
|
|
||||||
contents: string
|
|
||||||
password: boolean
|
|
||||||
views?: number
|
|
||||||
expiration?: number
|
|
||||||
}
|
|
||||||
export type NoteInfo = Pick<Note, 'password'>
|
|
||||||
export type NotePublic = Pick<Note, 'contents'>
|
|
||||||
|
|
||||||
export async function create(note: Note) {
|
|
||||||
const { data } = await base({
|
|
||||||
url: '/api/notes',
|
|
||||||
method: 'post',
|
|
||||||
data: note,
|
|
||||||
})
|
|
||||||
return data as { id: string }
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function get(id: string) {
|
|
||||||
const { data } = await base({
|
|
||||||
url: `/api/notes/${id}`,
|
|
||||||
method: 'delete',
|
|
||||||
})
|
|
||||||
return data as NotePublic
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function info(id: string) {
|
|
||||||
const { data } = await base({
|
|
||||||
url: `/api/notes/${id}`,
|
|
||||||
method: 'get',
|
|
||||||
})
|
|
||||||
return data as NoteInfo
|
|
||||||
}
|
|
@@ -1,33 +0,0 @@
|
|||||||
<script lang="ts">
|
|
||||||
import { onMount } from 'svelte'
|
|
||||||
export let icon: string
|
|
||||||
|
|
||||||
$: src = `/icons/${icon}.svg`
|
|
||||||
|
|
||||||
let html = null
|
|
||||||
|
|
||||||
onMount(async () => {
|
|
||||||
html = await fetch(src).then((res) => res.text())
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
{#if html === null}
|
|
||||||
<img on:click {...$$restProps} {src} alt={icon} />
|
|
||||||
{:else}
|
|
||||||
<div on:click {...$$restProps}>
|
|
||||||
{@html html}
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<style>
|
|
||||||
img,
|
|
||||||
div {
|
|
||||||
display: inline-block;
|
|
||||||
contain: strict;
|
|
||||||
box-sizing: content-box;
|
|
||||||
}
|
|
||||||
div > :global(svg) {
|
|
||||||
display: block;
|
|
||||||
fill: currentColor;
|
|
||||||
}
|
|
||||||
</style>
|
|
@@ -1,34 +0,0 @@
|
|||||||
<script lang="ts">
|
|
||||||
export let label: string = ''
|
|
||||||
export let value: string
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<label>
|
|
||||||
<small>
|
|
||||||
{label}
|
|
||||||
</small>
|
|
||||||
<textarea {...$$restProps} bind:value />
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
textarea {
|
|
||||||
width: 100%;
|
|
||||||
min-height: calc(100vh - 30rem);
|
|
||||||
margin: 0;
|
|
||||||
border: 2px solid var(--ui-bg-1);
|
|
||||||
resize: vertical;
|
|
||||||
outline: none;
|
|
||||||
padding: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (max-width: 30rem) {
|
|
||||||
textarea {
|
|
||||||
min-height: calc(100vh - 25rem);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
textarea:hover,
|
|
||||||
textarea:focus {
|
|
||||||
border-color: var(--ui-clr-primary);
|
|
||||||
}
|
|
||||||
</style>
|
|
@@ -1,159 +0,0 @@
|
|||||||
<script lang="ts">
|
|
||||||
import type { Note } from '$lib/api'
|
|
||||||
import { create } from '$lib/api'
|
|
||||||
import { getKeyFromString, encrypt } from '$lib/crypto'
|
|
||||||
|
|
||||||
import Button from '$lib/ui/Button.svelte'
|
|
||||||
import Switch from '$lib/ui/Switch.svelte'
|
|
||||||
import TextArea from '$lib/ui/TextArea.svelte'
|
|
||||||
import TextInput from '$lib/ui/TextInput.svelte'
|
|
||||||
|
|
||||||
let note: Note = {
|
|
||||||
contents: '',
|
|
||||||
password: false,
|
|
||||||
views: 1,
|
|
||||||
expiration: 60,
|
|
||||||
}
|
|
||||||
let password: string = ''
|
|
||||||
let result: { password: string; id: string } | null = null
|
|
||||||
let advanced = false
|
|
||||||
let type = false
|
|
||||||
let message = ''
|
|
||||||
let loading = false
|
|
||||||
let error: string | null = null
|
|
||||||
|
|
||||||
$: if (!advanced) {
|
|
||||||
note.views = 1
|
|
||||||
type = false
|
|
||||||
}
|
|
||||||
|
|
||||||
$: {
|
|
||||||
let fraction: string
|
|
||||||
fraction = type ? `${note.expiration} minutes` : `${note.views} views`
|
|
||||||
message = 'the note will expire and be destroyed after ' + fraction
|
|
||||||
}
|
|
||||||
|
|
||||||
async function submit() {
|
|
||||||
try {
|
|
||||||
error = null
|
|
||||||
loading = true
|
|
||||||
const data: Note = {
|
|
||||||
contents: note.contents,
|
|
||||||
password: !!password,
|
|
||||||
}
|
|
||||||
// @ts-ignore
|
|
||||||
if (type) data.expiration = parseInt(note.expiration)
|
|
||||||
// @ts-ignore
|
|
||||||
else data.views = parseInt(note.views)
|
|
||||||
if (data.password) {
|
|
||||||
const key = await getKeyFromString(password)
|
|
||||||
data.contents = await encrypt(data.contents, key)
|
|
||||||
}
|
|
||||||
|
|
||||||
const response = await create(data)
|
|
||||||
result = {
|
|
||||||
password: password,
|
|
||||||
id: response.id,
|
|
||||||
}
|
|
||||||
} catch {
|
|
||||||
error = 'could not create note.'
|
|
||||||
} finally {
|
|
||||||
loading = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function reset() {
|
|
||||||
window.location.reload()
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
{#if result}
|
|
||||||
{#if result.password}
|
|
||||||
<TextInput type="password" readonly value={result.password} copy />
|
|
||||||
<br />
|
|
||||||
{/if}
|
|
||||||
<TextInput type="text" readonly value="{window.location.origin}/note/{result.id}" copy />
|
|
||||||
<br />
|
|
||||||
<Button on:click={reset}>new</Button>
|
|
||||||
{:else}
|
|
||||||
<form on:submit|preventDefault={submit}>
|
|
||||||
<fieldset disabled={loading}>
|
|
||||||
<TextArea label="note" bind:value={note.contents} placeholder="..." />
|
|
||||||
|
|
||||||
<div class="bottom">
|
|
||||||
<Switch label="advanced" bind:value={advanced} />
|
|
||||||
<Button type="submit">create</Button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{#if error}
|
|
||||||
<div class="error-text">{error}</div>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<p><br />{message}</p>
|
|
||||||
|
|
||||||
<div class="advanced" class:hidden={!advanced}>
|
|
||||||
<br />
|
|
||||||
<div class="fields">
|
|
||||||
<TextInput
|
|
||||||
type="number"
|
|
||||||
label="views"
|
|
||||||
bind:value={note.views}
|
|
||||||
disabled={type}
|
|
||||||
max={100}
|
|
||||||
/>
|
|
||||||
<div class="middle-switch">
|
|
||||||
<Switch label="mode" bind:value={type} color={false} />
|
|
||||||
</div>
|
|
||||||
<TextInput
|
|
||||||
type="number"
|
|
||||||
label="minutes"
|
|
||||||
bind:value={note.expiration}
|
|
||||||
disabled={!type}
|
|
||||||
max={360}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<br />
|
|
||||||
<TextInput
|
|
||||||
type="password"
|
|
||||||
label="password"
|
|
||||||
placeholder="optional"
|
|
||||||
bind:value={password}
|
|
||||||
copy
|
|
||||||
random
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.fields {
|
|
||||||
display: flex;
|
|
||||||
}
|
|
||||||
.spacer {
|
|
||||||
width: 3rem;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
</fieldset>
|
|
||||||
</form>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.bottom {
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
align-items: flex-end;
|
|
||||||
margin-top: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.middle-switch {
|
|
||||||
margin: 0 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.advanced {
|
|
||||||
max-height: 14em;
|
|
||||||
overflow: hidden;
|
|
||||||
transition: var(--ui-anim);
|
|
||||||
}
|
|
||||||
|
|
||||||
.advanced.hidden {
|
|
||||||
max-height: 0;
|
|
||||||
}
|
|
||||||
</style>
|
|
@@ -1 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?><!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"><svg width="100%" height="100%" viewBox="0 0 475 200" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" xmlns:serif="http://www.serif.com/" style="fill-rule:evenodd;clip-rule:evenodd;stroke-linejoin:round;stroke-miterlimit:2;"><g id="Logo"><clipPath id="_clip1"><rect x="4.516" y="3.225" width="193.55" height="193.55"/></clipPath><g clip-path="url(#_clip1)"><g><g><path d="M171.604,43.296c-2.087,-0 -3.78,1.693 -3.78,3.78c-0,2.087 1.693,3.78 3.78,3.78c2.087,0 3.78,-1.693 3.78,-3.78c0,-2.087 -1.693,-3.78 -3.78,-3.78Z" style="fill-rule:nonzero;"/></g></g><g><g><path d="M101.291,134.023c-2.087,-0 -3.78,1.693 -3.78,3.78c-0,2.087 1.693,3.78 3.78,3.78c2.087,0 3.78,-1.693 3.78,-3.78c0,-2.087 -1.693,-3.78 -3.78,-3.78Z" style="fill-rule:nonzero;"/></g></g><g><g><path d="M154.215,96.22c-2.088,-0 -3.78,1.692 -3.78,3.78c-0,18.76 -15.263,34.023 -34.023,34.023c-2.088,-0 -3.78,1.692 -3.78,3.78c-0,2.088 1.692,3.78 3.78,3.78c22.929,0 41.583,-18.654 41.583,-41.583c0,-2.088 -1.692,-3.78 -3.78,-3.78Z" style="fill-rule:nonzero;"/></g></g><g><g><path d="M197.667,60.507l-9.515,-19.026c-4.101,-8.207 -12.349,-13.306 -21.526,-13.306c-7.48,-0 -14.627,3.547 -19.155,9.498l-10.678,12.016c-2.847,-4.047 -7.021,-7.049 -11.831,-8.421l-19.102,-5.459c-14.623,-4.178 -28.92,-15.441 -39.227,-30.901c-0.924,-1.386 -2.646,-2.003 -4.24,-1.521c-1.595,0.483 -2.685,1.953 -2.685,3.618l-0,20.372c-0,9.468 1.418,18.804 4.219,27.813c-2.936,0.73 -5.896,1.34 -8.842,1.816c-5.773,0.936 -11.654,1.411 -17.48,1.411l-29.309,-0c-1.374,-0 -2.64,0.746 -3.306,1.948c-0.666,1.202 -0.628,2.671 0.101,3.836l22.636,36.219c9.672,15.473 26.33,25.183 44.578,25.983l-36.36,41.158c-5.602,5.728 -3.654,15.315 3.746,18.396l20.018,9.887c0.088,0.044 0.179,0.084 0.271,0.121c5.789,2.313 12.389,-0.496 14.725,-6.279l13.969,-32.982l27.738,0c31.966,0 58.972,-25.967 58.972,-56.704l0,-22.682c0,-6.253 5.088,-11.341 11.341,-11.341l7.561,0c1.31,0 2.527,-0.678 3.216,-1.793c0.688,-1.114 0.751,-2.506 0.165,-3.677Zm-130.399,-42.236c10.418,12.203 23.307,21.034 36.515,24.808l19.104,5.459c3.725,1.063 6.865,3.624 8.666,7.048l-10.049,11.307l-17.652,-10.591c-7.646,-4.588 -16.746,-6.412 -25.776,-4.951c-2.837,0.46 -4.648,1.038 -6.877,1.759c-2.609,-8.332 -3.931,-16.971 -3.931,-25.733l0,-9.106Zm119.457,40.146c-10.422,-0 -18.901,8.479 -18.901,18.901l-0,22.682c-0,26.639 -23.544,49.144 -51.412,49.144l-30.242,-0c-10.771,-0 -20.452,5.983 -25.265,15.615l-0.798,1.596c-0.934,1.867 -0.177,4.137 1.691,5.071c1.867,0.934 4.138,0.176 5.072,-1.691c0.44,-0.586 3.306,-9.102 13.21,-12.125l-12.349,29.159c-0.01,0.024 -0.02,0.047 -0.03,0.071c-0.75,1.877 -2.864,2.851 -4.8,2.148c-21.279,-10.506 -19.997,-9.888 -20.252,-9.99c-2.526,-1.01 -3.191,-4.259 -1.267,-6.182c0.131,-0.131 8.026,-9.078 41.009,-46.411c13.867,-0.617 26.842,-6.319 36.694,-16.172c1.477,-1.476 1.477,-3.87 0,-5.346c-1.476,-1.477 -3.869,-1.476 -5.346,-0c-16.827,16.828 -36.803,13.634 -39.027,14.014c-16.604,0 -31.772,-8.407 -40.574,-22.488l-2.417,-3.868l2.729,1.065c17.308,6.753 38.919,4.347 53.817,-5.586c1.737,-1.158 2.206,-3.505 1.048,-5.242c-1.158,-1.737 -3.505,-2.207 -5.243,-1.048c-13.085,8.724 -31.922,10.666 -46.874,4.832l-12.185,-4.753l-9.896,-15.836l22.488,0c6.23,0 12.518,-0.507 18.688,-1.507c12.711,-2.055 18.051,-4.855 22.993,-5.655c7.181,-1.163 14.516,0.274 20.677,3.97l29.625,17.775c1.79,1.074 4.112,0.494 5.187,-1.296c1.074,-1.79 0.494,-4.113 -1.296,-5.187l-5.377,-3.226c26.559,-29.893 25.139,-28.272 25.319,-28.511c3.102,-4.136 8.038,-6.605 13.205,-6.605c6.293,0 11.95,3.497 14.764,9.127l6.779,13.555l-1.444,-0Z" style="fill-rule:nonzero;"/></g></g></g><text x="195.418px" y="127.131px" style="font-family:'Sofia-Regular', 'Sofia';font-size:60.681px;">cryptogeon</text></g></svg>
|
|
Before Width: | Height: | Size: 3.8 KiB |
@@ -1,85 +0,0 @@
|
|||||||
<script lang="ts">
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<header>
|
|
||||||
<a href="/">
|
|
||||||
<svg
|
|
||||||
width="100%"
|
|
||||||
height="100%"
|
|
||||||
viewBox="0 0 475 200"
|
|
||||||
version="1.1"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
xmlns:xlink="http://www.w3.org/1999/xlink"
|
|
||||||
xml:space="preserve"
|
|
||||||
style="fill-rule:evenodd;clip-rule:evenodd;stroke-linejoin:round;stroke-miterlimit:2;"
|
|
||||||
fill="currentColor"
|
|
||||||
><g id="Logo"
|
|
||||||
><clipPath id="_clip1"><rect x="4.516" y="3.225" width="193.55" height="193.55" /></clipPath
|
|
||||||
><g clip-path="url(#_clip1)"
|
|
||||||
><g
|
|
||||||
><g
|
|
||||||
><path
|
|
||||||
d="M171.604,43.296c-2.087,-0 -3.78,1.693 -3.78,3.78c-0,2.087 1.693,3.78 3.78,3.78c2.087,0 3.78,-1.693 3.78,-3.78c0,-2.087 -1.693,-3.78 -3.78,-3.78Z"
|
|
||||||
style="fill-rule:nonzero;"
|
|
||||||
/></g
|
|
||||||
></g
|
|
||||||
><g
|
|
||||||
><g
|
|
||||||
><path
|
|
||||||
d="M101.291,134.023c-2.087,-0 -3.78,1.693 -3.78,3.78c-0,2.087 1.693,3.78 3.78,3.78c2.087,0 3.78,-1.693 3.78,-3.78c0,-2.087 -1.693,-3.78 -3.78,-3.78Z"
|
|
||||||
style="fill-rule:nonzero;"
|
|
||||||
/></g
|
|
||||||
></g
|
|
||||||
><g
|
|
||||||
><g
|
|
||||||
><path
|
|
||||||
d="M154.215,96.22c-2.088,-0 -3.78,1.692 -3.78,3.78c-0,18.76 -15.263,34.023 -34.023,34.023c-2.088,-0 -3.78,1.692 -3.78,3.78c-0,2.088 1.692,3.78 3.78,3.78c22.929,0 41.583,-18.654 41.583,-41.583c0,-2.088 -1.692,-3.78 -3.78,-3.78Z"
|
|
||||||
style="fill-rule:nonzero;"
|
|
||||||
/></g
|
|
||||||
></g
|
|
||||||
><g
|
|
||||||
><g
|
|
||||||
><path
|
|
||||||
d="M197.667,60.507l-9.515,-19.026c-4.101,-8.207 -12.349,-13.306 -21.526,-13.306c-7.48,-0 -14.627,3.547 -19.155,9.498l-10.678,12.016c-2.847,-4.047 -7.021,-7.049 -11.831,-8.421l-19.102,-5.459c-14.623,-4.178 -28.92,-15.441 -39.227,-30.901c-0.924,-1.386 -2.646,-2.003 -4.24,-1.521c-1.595,0.483 -2.685,1.953 -2.685,3.618l-0,20.372c-0,9.468 1.418,18.804 4.219,27.813c-2.936,0.73 -5.896,1.34 -8.842,1.816c-5.773,0.936 -11.654,1.411 -17.48,1.411l-29.309,-0c-1.374,-0 -2.64,0.746 -3.306,1.948c-0.666,1.202 -0.628,2.671 0.101,3.836l22.636,36.219c9.672,15.473 26.33,25.183 44.578,25.983l-36.36,41.158c-5.602,5.728 -3.654,15.315 3.746,18.396l20.018,9.887c0.088,0.044 0.179,0.084 0.271,0.121c5.789,2.313 12.389,-0.496 14.725,-6.279l13.969,-32.982l27.738,0c31.966,0 58.972,-25.967 58.972,-56.704l0,-22.682c0,-6.253 5.088,-11.341 11.341,-11.341l7.561,0c1.31,0 2.527,-0.678 3.216,-1.793c0.688,-1.114 0.751,-2.506 0.165,-3.677Zm-130.399,-42.236c10.418,12.203 23.307,21.034 36.515,24.808l19.104,5.459c3.725,1.063 6.865,3.624 8.666,7.048l-10.049,11.307l-17.652,-10.591c-7.646,-4.588 -16.746,-6.412 -25.776,-4.951c-2.837,0.46 -4.648,1.038 -6.877,1.759c-2.609,-8.332 -3.931,-16.971 -3.931,-25.733l0,-9.106Zm119.457,40.146c-10.422,-0 -18.901,8.479 -18.901,18.901l-0,22.682c-0,26.639 -23.544,49.144 -51.412,49.144l-30.242,-0c-10.771,-0 -20.452,5.983 -25.265,15.615l-0.798,1.596c-0.934,1.867 -0.177,4.137 1.691,5.071c1.867,0.934 4.138,0.176 5.072,-1.691c0.44,-0.586 3.306,-9.102 13.21,-12.125l-12.349,29.159c-0.01,0.024 -0.02,0.047 -0.03,0.071c-0.75,1.877 -2.864,2.851 -4.8,2.148c-21.279,-10.506 -19.997,-9.888 -20.252,-9.99c-2.526,-1.01 -3.191,-4.259 -1.267,-6.182c0.131,-0.131 8.026,-9.078 41.009,-46.411c13.867,-0.617 26.842,-6.319 36.694,-16.172c1.477,-1.476 1.477,-3.87 0,-5.346c-1.476,-1.477 -3.869,-1.476 -5.346,-0c-16.827,16.828 -36.803,13.634 -39.027,14.014c-16.604,0 -31.772,-8.407 -40.574,-22.488l-2.417,-3.868l2.729,1.065c17.308,6.753 38.919,4.347 53.817,-5.586c1.737,-1.158 2.206,-3.505 1.048,-5.242c-1.158,-1.737 -3.505,-2.207 -5.243,-1.048c-13.085,8.724 -31.922,10.666 -46.874,4.832l-12.185,-4.753l-9.896,-15.836l22.488,0c6.23,0 12.518,-0.507 18.688,-1.507c12.711,-2.055 18.051,-4.855 22.993,-5.655c7.181,-1.163 14.516,0.274 20.677,3.97l29.625,17.775c1.79,1.074 4.112,0.494 5.187,-1.296c1.074,-1.79 0.494,-4.113 -1.296,-5.187l-5.377,-3.226c26.559,-29.893 25.139,-28.272 25.319,-28.511c3.102,-4.136 8.038,-6.605 13.205,-6.605c6.293,0 11.95,3.497 14.764,9.127l6.779,13.555l-1.444,-0Z"
|
|
||||||
style="fill-rule:nonzero;"
|
|
||||||
/></g
|
|
||||||
></g
|
|
||||||
></g
|
|
||||||
><text
|
|
||||||
x="195.418px"
|
|
||||||
y="127.131px"
|
|
||||||
style="font-family:'Sofia-Regular', 'Sofia';font-size:60.681px;">cryptogeon</text
|
|
||||||
></g
|
|
||||||
></svg
|
|
||||||
>
|
|
||||||
</a>
|
|
||||||
</header>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
a {
|
|
||||||
border: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
header {
|
|
||||||
text-align: center;
|
|
||||||
margin-top: 4rem;
|
|
||||||
margin-bottom: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media screen and (max-width: 30rem) {
|
|
||||||
header {
|
|
||||||
margin-top: 1rem;
|
|
||||||
margin-bottom: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
header svg {
|
|
||||||
max-height: 4rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
header svg {
|
|
||||||
width: 100%;
|
|
||||||
max-width: 16rem;
|
|
||||||
transform: translateX(-1rem);
|
|
||||||
}
|
|
||||||
</style>
|
|
@@ -1,45 +0,0 @@
|
|||||||
<script lang="ts">
|
|
||||||
import '../app.css'
|
|
||||||
import Header from '$lib/views/Header/index.svelte'
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<svelte:head>
|
|
||||||
<title>cryptgeon</title>
|
|
||||||
</svelte:head>
|
|
||||||
|
|
||||||
<main>
|
|
||||||
<Header />
|
|
||||||
<slot />
|
|
||||||
</main>
|
|
||||||
|
|
||||||
<footer>
|
|
||||||
<a href="/">/home</a>
|
|
||||||
<a href="/about">/about</a>
|
|
||||||
<a href="https://github.com/cupcakearmy/cryptgeon" target="_blank" rel="noopener">/code</a>
|
|
||||||
</footer>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
a {
|
|
||||||
margin: 0 0.5rem;
|
|
||||||
}
|
|
||||||
main {
|
|
||||||
padding: 1rem;
|
|
||||||
padding-bottom: 4rem;
|
|
||||||
width: 100%;
|
|
||||||
max-width: 35rem;
|
|
||||||
margin: 0 auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
footer {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
justify-content: flex-end;
|
|
||||||
align-items: center;
|
|
||||||
padding: 1rem;
|
|
||||||
position: fixed;
|
|
||||||
bottom: 0;
|
|
||||||
right: 0;
|
|
||||||
width: 100%;
|
|
||||||
background-color: var(--ui-bg-0-85);
|
|
||||||
}
|
|
||||||
</style>
|
|
@@ -1,66 +0,0 @@
|
|||||||
<script context="module">
|
|
||||||
import { browser, dev } from '$app/env'
|
|
||||||
export const hydrate = dev
|
|
||||||
export const router = browser
|
|
||||||
export const prerender = true
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<svelte:head>
|
|
||||||
<title>About</title>
|
|
||||||
</svelte:head>
|
|
||||||
|
|
||||||
<section class="content">
|
|
||||||
<h1>About</h1>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<i>cryptgeon</i> is an secure, open source sharing note service inspired by
|
|
||||||
<a href="https://privnote.com"><i>PrivNote</i></a>.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<b>▶ how does it work?</b>
|
|
||||||
<br />
|
|
||||||
each note has a 512bit generated <i>id</i> that is used to retrieve the note. data is stored in memory
|
|
||||||
and never persisted to disk.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<b>▶ Features</b>
|
|
||||||
<ul>
|
|
||||||
<li>view and time constrains</li>
|
|
||||||
<li>in memory, no persistence</li>
|
|
||||||
<li>in browser encryption → server cannot decrypt contents</li>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<b>▶ tech stack</b>
|
|
||||||
<br />
|
|
||||||
the backend is written in rust and the frontend is svelte and typescript.
|
|
||||||
<br />
|
|
||||||
you are welcomed to check & audit the
|
|
||||||
<a href="https://github.com/cupcakearmy/cryptgeon" target="_blank" rel="noopener">source code</a
|
|
||||||
>.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<br />
|
|
||||||
<b>▶ attributions</b>
|
|
||||||
<br />
|
|
||||||
<small>
|
|
||||||
icons made by <a href="https://www.freepik.com" title="Freepik">freepik</a> from
|
|
||||||
<a href="https://www.flaticon.com/" title="Flaticon">www.flaticon.com</a>
|
|
||||||
</small>
|
|
||||||
</p>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
section {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
ul {
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
padding-left: 1rem;
|
|
||||||
list-style: square;
|
|
||||||
}
|
|
||||||
</style>
|
|
@@ -1,98 +0,0 @@
|
|||||||
<script context="module" lang="ts">
|
|
||||||
export async function load({ page }) {
|
|
||||||
return {
|
|
||||||
props: page.params,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<script lang="ts">
|
|
||||||
import type { NotePublic } from '$lib/api'
|
|
||||||
import { info, get } from '$lib/api'
|
|
||||||
import { decrypt, getKeyFromString } from '$lib/crypto'
|
|
||||||
import Button from '$lib/ui/Button.svelte'
|
|
||||||
import TextInput from '$lib/ui/TextInput.svelte'
|
|
||||||
import copy from 'copy-to-clipboard'
|
|
||||||
|
|
||||||
import { onMount } from 'svelte'
|
|
||||||
|
|
||||||
export let id: string
|
|
||||||
let needPassword = false
|
|
||||||
let password: string = ''
|
|
||||||
let note: NotePublic | null = null
|
|
||||||
let exists = false
|
|
||||||
|
|
||||||
let loading = true
|
|
||||||
let error = false
|
|
||||||
|
|
||||||
onMount(async () => {
|
|
||||||
try {
|
|
||||||
loading = true
|
|
||||||
error = null
|
|
||||||
const data = await info(id)
|
|
||||||
needPassword = data.password
|
|
||||||
exists = true
|
|
||||||
} catch {
|
|
||||||
exists = false
|
|
||||||
} finally {
|
|
||||||
loading = false
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
async function show() {
|
|
||||||
const data = note || (await get(id)) // Don't get the content twice on wrong password.
|
|
||||||
if (needPassword) {
|
|
||||||
try {
|
|
||||||
const key = await getKeyFromString(password)
|
|
||||||
data.contents = await decrypt(data.contents, key)
|
|
||||||
error = false
|
|
||||||
} catch {
|
|
||||||
error = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
note = data
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
{#if !loading}
|
|
||||||
{#if !exists}
|
|
||||||
<p class="error-text">note was not found or was already deleted.</p>
|
|
||||||
{:else if note && !error}
|
|
||||||
<p class="error-text">you will not get the chance to see the note again.</p>
|
|
||||||
<div class="note">
|
|
||||||
{note.contents}
|
|
||||||
</div>
|
|
||||||
<br />
|
|
||||||
<Button on:click={() => copy(note.contents)}>copy to clipboard</Button>
|
|
||||||
{:else}
|
|
||||||
<form on:submit|preventDefault={show}>
|
|
||||||
<p>click below to show and delete the note if the counter has reached it's limit</p>
|
|
||||||
{#if needPassword}
|
|
||||||
<TextInput type="password" label="password" bind:value={password} />
|
|
||||||
<br />
|
|
||||||
{/if}
|
|
||||||
<Button type="submit">show note</Button>
|
|
||||||
{#if error}
|
|
||||||
<br />
|
|
||||||
<p class="error-text">
|
|
||||||
wrong password. could not decipher.
|
|
||||||
<br />
|
|
||||||
note already destroyed. try again without reloading the page.
|
|
||||||
</p>
|
|
||||||
{/if}
|
|
||||||
</form>
|
|
||||||
{/if}
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.note {
|
|
||||||
width: 100%;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
border: 2px solid var(--ui-bg-1);
|
|
||||||
outline: none;
|
|
||||||
padding: 0.5rem;
|
|
||||||
white-space: pre;
|
|
||||||
overflow: auto;
|
|
||||||
}
|
|
||||||
</style>
|
|
@@ -1 +0,0 @@
|
|||||||
<svg xmlns='http://www.w3.org/2000/svg' class='ionicon' viewBox='0 0 512 512'><title>Copy</title><path d='M456 480H136a24 24 0 01-24-24V128a16 16 0 0116-16h328a24 24 0 0124 24v320a24 24 0 01-24 24z'/><path d='M112 80h288V56a24 24 0 00-24-24H60a28 28 0 00-28 28v316a24 24 0 0024 24h24V112a32 32 0 0132-32z'/></svg>
|
|
Before Width: | Height: | Size: 313 B |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns='http://www.w3.org/2000/svg' class='ionicon' viewBox='0 0 512 512'><title>Dice</title><path d='M48 366.92L240 480V284L48 170zM192 288c8.84 0 16 10.75 16 24s-7.16 24-16 24-16-10.75-16-24 7.16-24 16-24zm-96 32c8.84 0 16 10.75 16 24s-7.16 24-16 24-16-10.75-16-24 7.16-24 16-24zM272 284v196l192-113.08V170zm48 140c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm0-88c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm96 32c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm0-88c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm32 77.64zM256 32L64 144l192 112 192-112zm0 120c-13.25 0-24-7.16-24-16s10.75-16 24-16 24 7.16 24 16-10.75 16-24 16z'/></svg>
|
|
Before Width: | Height: | Size: 728 B |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns='http://www.w3.org/2000/svg' class='ionicon' viewBox='0 0 512 512'><title>Eye Off</title><path d='M63.998 86.004l21.998-21.998L448 426.01l-21.998 21.998zM259.34 192.09l60.57 60.57a64.07 64.07 0 00-60.57-60.57zM252.66 319.91l-60.57-60.57a64.07 64.07 0 0060.57 60.57z'/><path d='M256 352a96 96 0 01-92.6-121.34l-69.07-69.08C66.12 187.42 39.24 221.14 16 256c26.42 44 62.56 89.24 100.2 115.18C159.38 400.92 206.33 416 255.76 416A233.47 233.47 0 00335 402.2l-53.61-53.6A95.84 95.84 0 01256 352zM256 160a96 96 0 0192.6 121.34L419.26 352c29.15-26.25 56.07-61.56 76.74-96-26.38-43.43-62.9-88.56-101.18-114.82C351.1 111.2 304.31 96 255.76 96a222.92 222.92 0 00-78.21 14.29l53.11 53.11A95.84 95.84 0 01256 160z'/></svg>
|
|
Before Width: | Height: | Size: 720 B |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns='http://www.w3.org/2000/svg' class='ionicon' viewBox='0 0 512 512'><title>Eye</title><circle cx='256' cy='256' r='64'/><path d='M394.82 141.18C351.1 111.2 304.31 96 255.76 96c-43.69 0-86.28 13-126.59 38.48C88.52 160.23 48.67 207 16 256c26.42 44 62.56 89.24 100.2 115.18C159.38 400.92 206.33 416 255.76 416c49 0 95.85-15.07 139.3-44.79C433.31 345 469.71 299.82 496 256c-26.38-43.43-62.9-88.56-101.18-114.82zM256 352a96 96 0 1196-96 96.11 96.11 0 01-96 96z'/></svg>
|
|
Before Width: | Height: | Size: 474 B |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns='http://www.w3.org/2000/svg' class='ionicon' viewBox='0 0 512 512'><title>Lock Closed</title><path d='M420 192h-68v-80a96 96 0 10-192 0v80H92a12 12 0 00-12 12v280a12 12 0 0012 12h328a12 12 0 0012-12V204a12 12 0 00-12-12zm-106 0H198v-80.75a58 58 0 11116 0z'/></svg>
|
|
Before Width: | Height: | Size: 275 B |
@@ -1,17 +0,0 @@
|
|||||||
const preprocess = require('svelte-preprocess')
|
|
||||||
const adapter = require('@sveltejs/adapter-static')
|
|
||||||
|
|
||||||
/** @type {import('@sveltejs/kit').Config} */
|
|
||||||
module.exports = {
|
|
||||||
// Consult https://github.com/sveltejs/svelte-preprocess
|
|
||||||
// for more information about preprocessors
|
|
||||||
preprocess: preprocess(),
|
|
||||||
|
|
||||||
kit: {
|
|
||||||
adapter: adapter({
|
|
||||||
fallback: 'index.html',
|
|
||||||
}),
|
|
||||||
// hydrate the <div id="svelte"> element in src/app.html
|
|
||||||
target: '#svelte',
|
|
||||||
},
|
|
||||||
}
|
|
@@ -1,30 +0,0 @@
|
|||||||
{
|
|
||||||
"compilerOptions": {
|
|
||||||
"moduleResolution": "node",
|
|
||||||
"module": "es2020",
|
|
||||||
"lib": ["es2020"],
|
|
||||||
"target": "es2019",
|
|
||||||
/**
|
|
||||||
svelte-preprocess cannot figure out whether you have a value or a type, so tell TypeScript
|
|
||||||
to enforce using \`import type\` instead of \`import\` for Types.
|
|
||||||
*/
|
|
||||||
"importsNotUsedAsValues": "error",
|
|
||||||
"isolatedModules": true,
|
|
||||||
"resolveJsonModule": true,
|
|
||||||
/**
|
|
||||||
To have warnings/errors of the Svelte compiler at the correct position,
|
|
||||||
enable source maps by default.
|
|
||||||
*/
|
|
||||||
"sourceMap": true,
|
|
||||||
"esModuleInterop": true,
|
|
||||||
"skipLibCheck": true,
|
|
||||||
"forceConsistentCasingInFileNames": true,
|
|
||||||
"baseUrl": ".",
|
|
||||||
"allowJs": true,
|
|
||||||
"checkJs": true,
|
|
||||||
"paths": {
|
|
||||||
"$lib/*": ["src/lib/*"]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"include": ["src/**/*.d.ts", "src/**/*.js", "src/**/*.ts", "src/**/*.svelte"]
|
|
||||||
}
|
|
Before Width: | Height: | Size: 70 KiB After Width: | Height: | Size: 76 KiB |
BIN
design/Logo.afdesign
(Stored with Git LFS)
Before Width: | Height: | Size: 3.8 KiB After Width: | Height: | Size: 12 KiB |
@@ -1 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?><!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"><svg width="100%" height="100%" viewBox="0 0 475 200" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" xmlns:serif="http://www.serif.com/" style="fill-rule:evenodd;clip-rule:evenodd;stroke-linejoin:round;stroke-miterlimit:2;"><path d="M475,20.713c0,-11.432 -9.281,-20.713 -20.713,-20.713l-433.574,0c-11.432,0 -20.713,9.281 -20.713,20.713l0,158.574c0,11.432 9.281,20.713 20.713,20.713l433.574,0c11.432,0 20.713,-9.281 20.713,-20.713l0,-158.574Z" style="fill:#fff;"/><g id="Logo"><clipPath id="_clip1"><rect x="29.237" y="13.493" width="173.013" height="173.013"/></clipPath><g clip-path="url(#_clip1)"><g><g><path d="M178.596,49.313c-1.865,-0 -3.379,1.513 -3.379,3.379c0,1.865 1.514,3.379 3.379,3.379c1.865,-0 3.379,-1.514 3.379,-3.379c0,-1.866 -1.514,-3.379 -3.379,-3.379Z" style="fill-rule:nonzero;"/></g></g><g><g><path d="M115.744,130.412c-1.866,0 -3.379,1.514 -3.379,3.38c-0,1.865 1.513,3.379 3.379,3.379c1.865,-0 3.379,-1.514 3.379,-3.379c-0,-1.866 -1.514,-3.38 -3.379,-3.38Z" style="fill-rule:nonzero;"/></g></g><g><g><path d="M163.052,96.621c-1.866,-0 -3.379,1.513 -3.379,3.379c-0,16.769 -13.643,30.412 -30.413,30.412c-1.866,0 -3.379,1.513 -3.379,3.38c0,1.866 1.513,3.379 3.379,3.379c20.496,-0 37.171,-16.675 37.171,-37.171c0,-1.866 -1.513,-3.379 -3.379,-3.379Z" style="fill-rule:nonzero;"/></g></g><g><g><path d="M201.894,64.697l-8.506,-17.006c-3.666,-7.337 -11.039,-11.895 -19.242,-11.895c-6.686,-0 -13.075,3.171 -17.122,8.49l-9.545,10.741c-2.545,-3.617 -6.277,-6.301 -10.576,-7.527l-17.075,-4.88c-13.071,-3.735 -25.851,-13.803 -35.064,-27.622c-0.826,-1.239 -2.366,-1.791 -3.791,-1.36c-1.425,0.432 -2.4,1.746 -2.4,3.235l-0,18.21c-0,8.463 1.268,16.809 3.771,24.862c-2.624,0.652 -5.27,1.198 -7.904,1.623c-5.16,0.837 -10.417,1.261 -15.625,1.261l-26.199,0c-1.228,0 -2.36,0.667 -2.955,1.741c-0.595,1.075 -0.561,2.388 0.09,3.429l20.235,32.376c8.646,13.832 23.535,22.511 39.847,23.226l-32.501,36.791c-5.008,5.121 -3.267,13.69 3.348,16.444l17.894,8.838c0.079,0.039 0.16,0.075 0.242,0.108c5.175,2.068 11.074,-0.443 13.163,-5.612l12.487,-29.483l24.794,0c28.575,0 52.715,-23.211 52.715,-50.687l0,-20.275c0,-5.59 4.548,-10.137 10.138,-10.137l6.758,-0c1.171,-0 2.259,-0.607 2.875,-1.603c0.616,-0.996 0.671,-2.241 0.148,-3.288Zm-116.563,-37.754c9.313,10.908 20.834,18.802 32.64,22.176l17.077,4.879c3.33,0.951 6.137,3.24 7.747,6.301l-8.983,10.107l-15.779,-9.467c-6.835,-4.101 -14.969,-5.732 -23.041,-4.426c-2.536,0.411 -4.155,0.927 -6.147,1.573c-2.332,-7.449 -3.514,-15.171 -3.514,-23.003l0,-8.14Zm106.782,35.886c-9.316,0 -16.896,7.58 -16.896,16.896l0,20.275c0,23.812 -21.045,43.929 -45.957,43.929l-27.033,0c-9.627,0 -18.281,5.349 -22.584,13.959l-0.713,1.426c-0.835,1.669 -0.158,3.699 1.511,4.533c1.67,0.835 3.699,0.158 4.534,-1.511c0.393,-0.524 2.955,-8.137 11.808,-10.839l-11.038,26.065c-0.01,0.021 -0.018,0.042 -0.027,0.064c-0.67,1.678 -2.56,2.548 -4.29,1.919c-19.022,-9.39 -17.875,-8.838 -18.104,-8.929c-2.258,-0.903 -2.852,-3.807 -1.132,-5.527c0.116,-0.116 7.174,-8.114 36.658,-41.486c12.395,-0.551 23.993,-5.648 32.8,-14.456c1.32,-1.319 1.32,-3.459 0,-4.779c-1.319,-1.32 -3.459,-1.319 -4.779,0c-15.042,15.043 -32.897,12.188 -34.886,12.528c-14.842,-0 -28.4,-7.515 -36.269,-20.103l-2.16,-3.457l2.439,0.952c15.471,6.036 34.79,3.886 48.107,-4.993c1.552,-1.035 1.972,-3.133 0.937,-4.686c-1.036,-1.553 -3.134,-1.972 -4.687,-0.937c-11.696,7.799 -28.535,9.535 -41.901,4.32l-10.892,-4.249l-8.846,-14.155l20.102,-0c5.57,-0 11.19,-0.454 16.706,-1.348c11.362,-1.837 16.135,-4.34 20.552,-5.055c6.42,-1.039 12.976,0.245 18.484,3.549l26.481,15.888c1.6,0.961 3.676,0.442 4.637,-1.158c0.96,-1.6 0.441,-3.676 -1.159,-4.636l-4.806,-2.884c23.741,-26.721 22.472,-25.272 22.633,-25.486c2.772,-3.697 7.185,-5.904 11.803,-5.904c5.626,0 10.683,3.126 13.198,8.159l6.059,12.116l-1.29,0Z" style="fill-rule:nonzero;"/></g></g></g><text x="199.883px" y="124.252px" style="font-family:'Sofia-Regular', 'Sofia';font-size:54.242px;">cryptogeon</text></g></svg>
|
|
Before Width: | Height: | Size: 4.0 KiB |
@@ -6,11 +6,16 @@ version: '3.7'
|
|||||||
services:
|
services:
|
||||||
memcached:
|
memcached:
|
||||||
image: memcached:1-alpine
|
image: memcached:1-alpine
|
||||||
entrypoint: memcached -m 128
|
restart: unless-stopped
|
||||||
|
entrypoint: memcached -m 256M -I 128M
|
||||||
ports:
|
ports:
|
||||||
- 11211:11211
|
- 11211:11211
|
||||||
|
|
||||||
app:
|
app:
|
||||||
build: .
|
build: .
|
||||||
|
depends_on:
|
||||||
|
- memcached
|
||||||
|
environment:
|
||||||
|
SIZE_LIMIT: 128M
|
||||||
ports:
|
ports:
|
||||||
- 80:5000
|
- 80:5000
|
||||||
|
11
entry.sh
Executable file
@@ -0,0 +1,11 @@
|
|||||||
|
#!/bin/ash
|
||||||
|
|
||||||
|
echo "Waiting for memcached"
|
||||||
|
|
||||||
|
while ! nc -z -w 1 memcached 11211; do
|
||||||
|
sleep 1
|
||||||
|
echo "..."
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "Starting server"
|
||||||
|
/app/cryptgeon
|
22
examples/nginx/docker-compose.yaml
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
services:
|
||||||
|
memcached:
|
||||||
|
image: memcached:1-alpine
|
||||||
|
entrypoint: memcached -m 256 -I 128 # Limit to 128 MB Ram, customize at free will. -m must be at least double than -I.
|
||||||
|
|
||||||
|
app:
|
||||||
|
image: cupcakearmy/cryptgeon:latest
|
||||||
|
depends_on:
|
||||||
|
- memcached
|
||||||
|
|
||||||
|
proxy:
|
||||||
|
image: nginx:alpine
|
||||||
|
depends_on:
|
||||||
|
- app
|
||||||
|
volumes:
|
||||||
|
- ./nginx-plain.conf:/etc/nginx/conf.d/default.conf
|
||||||
|
# Or with tls
|
||||||
|
# - ./nginx-tls.conf:/etc/nginx/conf.d/default.conf
|
||||||
|
ports:
|
||||||
|
- 80:80
|
13
examples/nginx/nginx-plain.conf
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
server {
|
||||||
|
listen 80;
|
||||||
|
listen [::]:80;
|
||||||
|
server_name _;
|
||||||
|
|
||||||
|
location / {
|
||||||
|
proxy_pass http://app:5000/;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
}
|
||||||
|
}
|
29
examples/nginx/nginx-tls.conf
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
# You should change the server_name to something sensible.
|
||||||
|
# Also you need to specify the path to the ssl certificates.
|
||||||
|
|
||||||
|
server {
|
||||||
|
listen 80;
|
||||||
|
listen [::]:80;
|
||||||
|
server_name _;
|
||||||
|
|
||||||
|
# Enforce HTTPS
|
||||||
|
return 301 https://$server_name$request_uri;
|
||||||
|
}
|
||||||
|
|
||||||
|
server {
|
||||||
|
listen 443 ssl http2;
|
||||||
|
listen [::]:443 ssl http2;
|
||||||
|
server_name _;
|
||||||
|
|
||||||
|
ssl_certificate /path/to/fullchain.pem;
|
||||||
|
ssl_certificate_key /path/to/privkey.pem;
|
||||||
|
ssl_trusted_certificate /path/to/fullchain.pem;
|
||||||
|
|
||||||
|
location / {
|
||||||
|
proxy_pass http://app:5000/;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
}
|
||||||
|
}
|
161
examples/scratch/README.md
Normal file
@@ -0,0 +1,161 @@
|
|||||||
|
# Install from scratch.
|
||||||
|
|
||||||
|
This is a tiny guide to install cryptgeon on (probably) any unix system (and maybe windows?) from scratch using traefik as the proxy, which will manage certificates and handle https for us.
|
||||||
|
|
||||||
|
1. Install Docker & Docker Compose.
|
||||||
|
2. Install Traefik.
|
||||||
|
3. Run the cryptgeon.
|
||||||
|
4. [Optional] install watchtower to keep up to date.
|
||||||
|
|
||||||
|
## Install Docker & DOcker Compose
|
||||||
|
|
||||||
|
- [Docker](https://docs.docker.com/engine/install/)
|
||||||
|
- [Compose](https://docs.docker.com/compose/install/)
|
||||||
|
|
||||||
|
## Install Traefik 2.0
|
||||||
|
|
||||||
|
[Traefik](https://doc.traefik.io/traefik/) is a router & proxy that makes deployment of containers incredibly easy. It will manage all the https certificates, routing, etc.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
/foo/bar/traefik/
|
||||||
|
├── docker-compose.yaml
|
||||||
|
└── traefik.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
# docker-compose.yaml
|
||||||
|
|
||||||
|
version: '3.8'
|
||||||
|
services:
|
||||||
|
traefik:
|
||||||
|
image: traefik:2.6
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- '80:80'
|
||||||
|
- '443:443'
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
- ./traefik.yaml:/etc/traefik/traefik.yaml:ro
|
||||||
|
- ./data:/data
|
||||||
|
labels:
|
||||||
|
- 'traefik.enable=true'
|
||||||
|
|
||||||
|
# HTTP to HTTPS redirection
|
||||||
|
- 'traefik.http.routers.http_catchall.rule=HostRegexp(`{any:.+}`)'
|
||||||
|
- 'traefik.http.routers.http_catchall.entrypoints=insecure'
|
||||||
|
- 'traefik.http.routers.http_catchall.middlewares=https_redirect'
|
||||||
|
- 'traefik.http.middlewares.https_redirect.redirectscheme.scheme=https'
|
||||||
|
- 'traefik.http.middlewares.https_redirect.redirectscheme.permanent=true'
|
||||||
|
|
||||||
|
networks:
|
||||||
|
default:
|
||||||
|
external: true
|
||||||
|
name: proxy
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
# traefik.yaml
|
||||||
|
|
||||||
|
api:
|
||||||
|
dashboard: true
|
||||||
|
|
||||||
|
# Define HTTP and HTTPS entrypoint
|
||||||
|
entryPoints:
|
||||||
|
insecure:
|
||||||
|
address: ':80'
|
||||||
|
secure:
|
||||||
|
address: ':443'
|
||||||
|
|
||||||
|
# Dynamic configuration will come from docker labels
|
||||||
|
providers:
|
||||||
|
docker:
|
||||||
|
endpoint: 'unix:///var/run/docker.sock'
|
||||||
|
network: 'proxy'
|
||||||
|
exposedByDefault: false
|
||||||
|
|
||||||
|
# Enable acme with http file challenge
|
||||||
|
certificatesResolvers:
|
||||||
|
le:
|
||||||
|
acme:
|
||||||
|
email: me@example.org
|
||||||
|
storage: /data/acme.json
|
||||||
|
httpChallenge:
|
||||||
|
entryPoint: insecure
|
||||||
|
```
|
||||||
|
|
||||||
|
**Run**
|
||||||
|
|
||||||
|
```sh
|
||||||
|
docker network create proxy
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
## Cryptgeon
|
||||||
|
|
||||||
|
Create another docker-compose.yaml file in another folder. We will assume that the domain is `cryptgeon.example.org`.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
/foo/bar/cryptgeon/
|
||||||
|
└── docker-compose.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
networks:
|
||||||
|
proxy:
|
||||||
|
external: true
|
||||||
|
|
||||||
|
services:
|
||||||
|
memcached:
|
||||||
|
image: memcached:1-alpine
|
||||||
|
restart: unless-stopped
|
||||||
|
entrypoint: memcached -m 256M -I 4M # Limit to 128 MB Ram, customize at free will.
|
||||||
|
|
||||||
|
app:
|
||||||
|
image: cupcakearmy/cryptgeon:latest
|
||||||
|
restart: unless-stopped
|
||||||
|
depends_on:
|
||||||
|
- memcached
|
||||||
|
environment:
|
||||||
|
SIZE_LIMIT: 4 MiB
|
||||||
|
networks:
|
||||||
|
- default
|
||||||
|
- proxy
|
||||||
|
labels:
|
||||||
|
- traefik.enable=true
|
||||||
|
- traefik.http.routers.cryptgeon.rule=Host(`cryptgeon.example.org`)
|
||||||
|
- traefik.http.routers.cryptgeon.entrypoints=secure
|
||||||
|
- traefik.http.routers.cryptgeon.tls.certresolver=le
|
||||||
|
```
|
||||||
|
|
||||||
|
**Run**
|
||||||
|
|
||||||
|
```sh
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
## Watchtower
|
||||||
|
|
||||||
|
> A container-based solution for automating Docker container base image updates.
|
||||||
|
|
||||||
|
[Watchtower](https://containrrr.dev/watchtower/) will keep our containers up to date. The interval is set to once a day and also configured to delete old images to prevent cluttering.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
/foo/bar/watchtower/
|
||||||
|
└── docker-compose.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
# docker-compose.yaml
|
||||||
|
|
||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
services:
|
||||||
|
watchtower:
|
||||||
|
image: containrrr/watchtower
|
||||||
|
restart: unless-stopped
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
command: --cleanup --interval 86400
|
||||||
|
```
|
1
client/.gitignore → frontend/.gitignore
vendored
@@ -1,5 +1,6 @@
|
|||||||
.DS_Store
|
.DS_Store
|
||||||
node_modules
|
node_modules
|
||||||
/.svelte
|
/.svelte
|
||||||
|
/.svelte-kit
|
||||||
/build
|
/build
|
||||||
/functions
|
/functions
|
18
frontend/README.md
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# Cryptgeon Frontend
|
||||||
|
|
||||||
|
## Locale
|
||||||
|
|
||||||
|
Download with these settings:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"format": "json",
|
||||||
|
"indentation": "tab",
|
||||||
|
"json_unescaped_slashes": true,
|
||||||
|
"export_sort": "first_added",
|
||||||
|
"original_filenames": false,
|
||||||
|
"export_empty_as": "skip",
|
||||||
|
"add_newline_eof": true,
|
||||||
|
"replace_breaks": false
|
||||||
|
}
|
||||||
|
```
|
@@ -1,4 +1,4 @@
|
|||||||
├─ MIT: 43
|
├─ MIT: 46
|
||||||
├─ MIT*: 2
|
├─ MIT*: 2
|
||||||
├─ BSD-3-Clause: 2
|
├─ BSD-3-Clause: 2
|
||||||
├─ ISC: 1
|
├─ ISC: 1
|
|
42
frontend/locales/de.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"common": {
|
||||||
|
"note": "Hinweis",
|
||||||
|
"file": "Datei",
|
||||||
|
"advanced": "erweitert",
|
||||||
|
"create": "erstellen",
|
||||||
|
"loading": "Läd...",
|
||||||
|
"mode": "Modus",
|
||||||
|
"views": "{n, plural, =0 {Ansichten} =1 {1 Ansicht} other {# Ansichten}}",
|
||||||
|
"minutes": "{n, plural, =0 {Minuten} =1 {1 Minute} other {# Minuten}}",
|
||||||
|
"max": "max",
|
||||||
|
"share_link": "Link teilen",
|
||||||
|
"copy_clipboard": "in die Zwischenablage kopieren"
|
||||||
|
},
|
||||||
|
"home": {
|
||||||
|
"intro": "Senden Sie ganz einfach <i>vollständig verschlüsselte</i>, sichere Notizen oder Dateien mit einem Klick. Erstellen Sie einfach eine Notiz und teilen Sie den Link.",
|
||||||
|
"explanation": "die Notiz verfällt und wird nach {type} zerstört.",
|
||||||
|
"new_note": "neue Note",
|
||||||
|
"new_note_notice": "<b>Verfügbarkeit:</b><br />es ist nicht garantiert, dass die Notiz gespeichert wird, da alles im Speicher gehalten wird. Wenn dieser voll ist, werden die ältesten Notizen entfernt.<br />(Sie werden wahrscheinlich keine Probleme haben, seien Sie nur gewarnt).",
|
||||||
|
"errors": {
|
||||||
|
"note_to_big": "Notiz konnte nicht erstellt werden. Notiz ist zu groß",
|
||||||
|
"note_error": "konnte keine Notiz erstellen. Bitte versuchen Sie es erneut.",
|
||||||
|
"max": "max: {n}",
|
||||||
|
"empty_content": "Notiz ist leer."
|
||||||
|
},
|
||||||
|
"copied_to_clipboard": "in die Zwischenablage kopiert 🔗"
|
||||||
|
},
|
||||||
|
"show": {
|
||||||
|
"errors": {
|
||||||
|
"not_found": "wurde nicht gefunden oder wurde bereits gelöscht.",
|
||||||
|
"decryption_failed": "falsches Passwort. konnte nicht entziffert werden. wahrscheinlich ein defekter Link. Notiz wurde zerstört."
|
||||||
|
},
|
||||||
|
"explanation": "Klicken Sie unten, um die Notiz anzuzeigen und zu löschen, wenn der Zähler sein Limit erreicht hat",
|
||||||
|
"show_note": "Notiz anzeigen",
|
||||||
|
"warning_will_not_see_again": "haben Sie <b>keine</b> Gelegenheit, die Notiz noch einmal zu sehen.",
|
||||||
|
"download_all": "alle herunterladen"
|
||||||
|
},
|
||||||
|
"file_upload": {
|
||||||
|
"selected_files": "Ausgewählte Dateien",
|
||||||
|
"no_files_selected": "Keine Dateien ausgewählt"
|
||||||
|
}
|
||||||
|
}
|
42
frontend/locales/en.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"common": {
|
||||||
|
"note": "note",
|
||||||
|
"file": "file",
|
||||||
|
"advanced": "advanced",
|
||||||
|
"create": "create",
|
||||||
|
"loading": "loading...",
|
||||||
|
"mode": "mode",
|
||||||
|
"views": "{n, plural, =0 {views} =1 {1 view} other {# views}}",
|
||||||
|
"minutes": "{n, plural, =0 {minutes} =1 {1 minute} other {# minutes}}",
|
||||||
|
"max": "max",
|
||||||
|
"share_link": "share link",
|
||||||
|
"copy_clipboard": "copy to clipboard"
|
||||||
|
},
|
||||||
|
"home": {
|
||||||
|
"intro": "Easily send <i>fully encrypted</i>, secure notes or files with one click. Just create a note and share the link.",
|
||||||
|
"explanation": "the note will expire and be destroyed after {type}.",
|
||||||
|
"new_note": "new note",
|
||||||
|
"new_note_notice": "<b>availability:</b><br />the note is not guaranteed to be stored as everything is kept in ram, if it fills up the oldest notes will be removed.<br />(you probably will be fine, just be warned.)",
|
||||||
|
"errors": {
|
||||||
|
"note_to_big": "could not create note. note is to big",
|
||||||
|
"note_error": "could not create note. please try again.",
|
||||||
|
"max": "max: {n}",
|
||||||
|
"empty_content": "note is empty."
|
||||||
|
},
|
||||||
|
"copied_to_clipboard": "copied to clipboard 🔗"
|
||||||
|
},
|
||||||
|
"show": {
|
||||||
|
"errors": {
|
||||||
|
"not_found": "note was not found or was already deleted.",
|
||||||
|
"decryption_failed": "wrong password. could not decipher. probably a broken link. note was destroyed."
|
||||||
|
},
|
||||||
|
"explanation": "click below to show and delete the note if the counter has reached it's limit",
|
||||||
|
"show_note": "show note",
|
||||||
|
"warning_will_not_see_again": "you will <b>not</b> get the chance to see the note again.",
|
||||||
|
"download_all": "download all"
|
||||||
|
},
|
||||||
|
"file_upload": {
|
||||||
|
"selected_files": "Selected Files",
|
||||||
|
"no_files_selected": "No Files Selected"
|
||||||
|
}
|
||||||
|
}
|
42
frontend/locales/es.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"common": {
|
||||||
|
"note": "nota",
|
||||||
|
"file": "archivo",
|
||||||
|
"advanced": "avanzado",
|
||||||
|
"create": "crear",
|
||||||
|
"loading": "cargando...",
|
||||||
|
"mode": "modo",
|
||||||
|
"views": "{n, plural, =0 {vistas} =1 {1 vista} other {# vistas}}",
|
||||||
|
"minutes": "{n, plural, =0 {minutos} =1 {1 minuto} other {# minutos}}",
|
||||||
|
"max": "max",
|
||||||
|
"share_link": "compartir enlace",
|
||||||
|
"copy_clipboard": "copiar al portapapeles"
|
||||||
|
},
|
||||||
|
"home": {
|
||||||
|
"intro": "Envía fácilmente notas o archivos <i>totalmente encriptados</i> y seguros con un solo clic. Solo tienes que crear una nota y compartir el enlace.",
|
||||||
|
"explanation": "la nota expirará y se destruirá después de {type}.",
|
||||||
|
"new_note": "nueva nota",
|
||||||
|
"new_note_notice": "<b>disponibilidad:</b><br />no se garantiza que la nota se almacene, ya que todo se guarda en la memoria RAM, si se llena se eliminarán las notas más antiguas.<br />(probablemente estará bien, sólo está advertido.)",
|
||||||
|
"errors": {
|
||||||
|
"note_to_big": "no se pudo crear la nota. la nota es demasiado grande",
|
||||||
|
"note_error": "No se ha podido crear la nota. Por favor, inténtelo de nuevo.",
|
||||||
|
"max": "max: {n}",
|
||||||
|
"empty_content": "la nota está vacía."
|
||||||
|
},
|
||||||
|
"copied_to_clipboard": "copiado al portapapeles 🔗"
|
||||||
|
},
|
||||||
|
"show": {
|
||||||
|
"errors": {
|
||||||
|
"not_found": "la nota no se encontró o ya fue borrada.",
|
||||||
|
"decryption_failed": "contraseña incorrecta. no se pudo descifrar. probablemente un enlace roto. la nota fue destruida."
|
||||||
|
},
|
||||||
|
"explanation": "pulse abajo para mostrar y borrar la nota si el contador ha llegado a su límite",
|
||||||
|
"show_note": "mostrar nota",
|
||||||
|
"warning_will_not_see_again": " <b>no</b> tendrás la oportunidad de volver a ver la nota.",
|
||||||
|
"download_all": "descargar todo"
|
||||||
|
},
|
||||||
|
"file_upload": {
|
||||||
|
"selected_files": "Archivos seleccionados",
|
||||||
|
"no_files_selected": "No hay archivos seleccionados"
|
||||||
|
}
|
||||||
|
}
|
42
frontend/locales/fr.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"common": {
|
||||||
|
"note": "note",
|
||||||
|
"file": "fichier",
|
||||||
|
"advanced": "avancé",
|
||||||
|
"create": "créer",
|
||||||
|
"loading": "chargement...",
|
||||||
|
"mode": "mode",
|
||||||
|
"views": "{n, plural, =0 {vues} =1 {1 vue} other {# vues}}",
|
||||||
|
"minutes": "{n, plural, =0 {minutes} =1 {1 minute} other {# minutes}}",
|
||||||
|
"max": "max",
|
||||||
|
"share_link": "partager le lien",
|
||||||
|
"copy_clipboard": "copier dans le presse-papiers"
|
||||||
|
},
|
||||||
|
"home": {
|
||||||
|
"intro": "Envoyez facilement des notes ou des fichiers <i>entièrement cryptés</i> et sécurisés en un seul clic. Il suffit de créer une note et de partager le lien.",
|
||||||
|
"explanation": "la note expirera et sera détruite après {type}.",
|
||||||
|
"new_note": "nouvelle note",
|
||||||
|
"new_note_notice": "<b>disponibilité :</b><br />la note n'est pas garantie d'être stockée car tout est conservé dans la mémoire vive, si elle se remplit les notes les plus anciennes seront supprimées.<br />(vous serez probablement bien, soyez juste averti.)",
|
||||||
|
"errors": {
|
||||||
|
"note_to_big": "Impossible de créer une note. La note est trop grande",
|
||||||
|
"note_error": "n'a pas pu créer de note. Veuillez réessayer.",
|
||||||
|
"max": "max: {n}",
|
||||||
|
"empty_content": "La note est vide."
|
||||||
|
},
|
||||||
|
"copied_to_clipboard": "copié dans le presse-papiers 🔗"
|
||||||
|
},
|
||||||
|
"show": {
|
||||||
|
"errors": {
|
||||||
|
"not_found": "La note n'a pas été trouvée ou a déjà été supprimée.",
|
||||||
|
"decryption_failed": "mauvais mot de passe. impossible à déchiffrer. probablement un lien brisé. la note a été détruite."
|
||||||
|
},
|
||||||
|
"explanation": "Cliquez ci-dessous pour afficher et supprimer la note si le compteur a atteint sa limite.",
|
||||||
|
"show_note": "note de présentation",
|
||||||
|
"warning_will_not_see_again": "vous <b>n'aurez pas</b> la chance de revoir la note.",
|
||||||
|
"download_all": "télécharger tout"
|
||||||
|
},
|
||||||
|
"file_upload": {
|
||||||
|
"selected_files": "Fichiers sélectionnés",
|
||||||
|
"no_files_selected": "Aucun fichier sélectionné"
|
||||||
|
}
|
||||||
|
}
|
42
frontend/locales/it.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"common": {
|
||||||
|
"note": "nota",
|
||||||
|
"file": "file",
|
||||||
|
"advanced": "avanzato",
|
||||||
|
"create": "crea",
|
||||||
|
"loading": "carica...",
|
||||||
|
"mode": "modalita",
|
||||||
|
"views": "{n, plural, =0 {viste} =1 {1 vista} other {# viste}}",
|
||||||
|
"minutes": "{n, plural, =0 {minuti} =1 {1 minuto} other {# minuti}}",
|
||||||
|
"max": "max",
|
||||||
|
"share_link": "condividi link",
|
||||||
|
"copy_clipboard": "copia negli appunti"
|
||||||
|
},
|
||||||
|
"home": {
|
||||||
|
"intro": "Invia facilmente note o file <i>completamente criptati</i> e sicuri con un solo clic. Basta creare una nota e condividere il link.",
|
||||||
|
"explanation": "la nota scadrà e sarà distrutta dopo {type}.",
|
||||||
|
"new_note": "nuova nota",
|
||||||
|
"new_note_notice": "<b>disponibilità:</b><br />la nota non è garantita per essere memorizzata come tutto è tenuto in ram, se si riempie le note più vecchie saranno rimosse.<br />(probabilmente andrà bene, basta essere avvertiti).",
|
||||||
|
"errors": {
|
||||||
|
"note_to_big": "impossibile creare una nota. la nota è troppo grande",
|
||||||
|
"note_error": "Impossibile creare la nota. Riprova.",
|
||||||
|
"max": "max: {n}",
|
||||||
|
"empty_content": "la nota è vuota."
|
||||||
|
},
|
||||||
|
"copied_to_clipboard": "copiato negli appunti 🔗"
|
||||||
|
},
|
||||||
|
"show": {
|
||||||
|
"errors": {
|
||||||
|
"not_found": "non è stata trovata o è stata già cancellata.",
|
||||||
|
"decryption_failed": "password sbagliata. non ha potuto decifrare. probabilmente un link rotto. la nota è stata distrutta."
|
||||||
|
},
|
||||||
|
"explanation": "clicca sotto per mostrare e cancellare la nota se il contatore ha raggiunto il suo limite",
|
||||||
|
"show_note": "mostra la nota",
|
||||||
|
"warning_will_not_see_again": " <b>non</b> avrete la possibilità di rivedere la nota.",
|
||||||
|
"download_all": "scarica tutti"
|
||||||
|
},
|
||||||
|
"file_upload": {
|
||||||
|
"selected_files": "File selezionati",
|
||||||
|
"no_files_selected": "Nessun file selezionato"
|
||||||
|
}
|
||||||
|
}
|
35
frontend/package.json
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
{
|
||||||
|
"private": true,
|
||||||
|
"scripts": {
|
||||||
|
"dev": "svelte-kit dev",
|
||||||
|
"build": "svelte-kit build",
|
||||||
|
"preview": "svelte-kit preview",
|
||||||
|
"check": "svelte-check --tsconfig tsconfig.json",
|
||||||
|
"licenses": "license-checker --summary > licenses.csv",
|
||||||
|
"locale:download": "node scripts/locale.js"
|
||||||
|
},
|
||||||
|
"type": "module",
|
||||||
|
"devDependencies": {
|
||||||
|
"@lokalise/node-api": "^7.2.0",
|
||||||
|
"@sveltejs/adapter-static": "^1.0.0-next.34",
|
||||||
|
"@sveltejs/kit": "^1.0.0-next.348",
|
||||||
|
"@types/file-saver": "^2.0.5",
|
||||||
|
"@types/sanitize-html": "^2.6.2",
|
||||||
|
"adm-zip": "^0.5.9",
|
||||||
|
"dotenv": "^16.0.1",
|
||||||
|
"svelte": "^3.48.0",
|
||||||
|
"svelte-check": "^2.7.2",
|
||||||
|
"svelte-intl-precompile": "^0.10.1",
|
||||||
|
"svelte-preprocess": "^4.10.7",
|
||||||
|
"tslib": "^2.4.0",
|
||||||
|
"typescript": "^4.7.3",
|
||||||
|
"vite": "^2.9.10"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@fontsource/fira-mono": "^4.5.8",
|
||||||
|
"copy-to-clipboard": "^3.3.1",
|
||||||
|
"file-saver": "^2.0.5",
|
||||||
|
"pretty-bytes": "^5.6.0",
|
||||||
|
"sanitize-html": "^2.7.0"
|
||||||
|
}
|
||||||
|
}
|
1644
frontend/pnpm-lock.yaml
generated
Normal file
54
frontend/scripts/locale.js
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import dotenv from 'dotenv'
|
||||||
|
import { LokaliseApi } from '@lokalise/node-api'
|
||||||
|
import https from 'https'
|
||||||
|
import AdmZip from 'adm-zip'
|
||||||
|
|
||||||
|
dotenv.config()
|
||||||
|
|
||||||
|
const apiKey = process.env.LOKALISE_API_KEY
|
||||||
|
const project_id = process.env.LOKALISE_PROJECT
|
||||||
|
if (!apiKey) throw new Error('No API Key set for Lokalize! Set with "LOKALISE_API_KEY"')
|
||||||
|
if (!project_id) throw new Error('No project id set for Lokalize! Set with "LOKALISE_PROJECT"')
|
||||||
|
const client = new LokaliseApi({ apiKey })
|
||||||
|
|
||||||
|
const WGet = (url) =>
|
||||||
|
new Promise((done) => {
|
||||||
|
https
|
||||||
|
.get(url, (res) => {
|
||||||
|
const data = []
|
||||||
|
res
|
||||||
|
.on('data', (chunk) => {
|
||||||
|
data.push(chunk)
|
||||||
|
})
|
||||||
|
.on('end', () => {
|
||||||
|
let buffer = Buffer.concat(data)
|
||||||
|
done(buffer)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.on('error', (err) => {
|
||||||
|
console.log('download error:', err)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
async function download() {
|
||||||
|
// For details see: https://app.lokalise.com/api2docs/curl/#transition-download-files-post
|
||||||
|
const download = await client.files().download(project_id, {
|
||||||
|
format: 'json',
|
||||||
|
indentation: 'tab',
|
||||||
|
json_unescaped_slashes: true,
|
||||||
|
original_filenames: false,
|
||||||
|
bundle_structure: '%LANG_ISO%.%FORMAT%',
|
||||||
|
export_sort: 'first_added',
|
||||||
|
export_empty_as: 'skip',
|
||||||
|
add_newline_eof: true,
|
||||||
|
replace_breaks: false,
|
||||||
|
})
|
||||||
|
const buffered = await WGet(download.bundle_url)
|
||||||
|
const zip = new AdmZip(buffered)
|
||||||
|
zip.extractAllTo('./locales', true)
|
||||||
|
}
|
||||||
|
|
||||||
|
download().catch((e) => {
|
||||||
|
console.error(e)
|
||||||
|
process.exit(1)
|
||||||
|
})
|
124
frontend/src/app.css
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
@import '@fontsource/fira-mono';
|
||||||
|
|
||||||
|
* {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
:root {
|
||||||
|
font-family: 'Fira Mono', monospace;
|
||||||
|
|
||||||
|
--ui-bg-0: #111;
|
||||||
|
--ui-bg-0-85: #111111d9;
|
||||||
|
--ui-bg-1: #333;
|
||||||
|
--ui-bg-2: #444;
|
||||||
|
--ui-text-0: #fefefe;
|
||||||
|
--ui-text-1: #eee;
|
||||||
|
--ui-clr-primary: hsl(186, 65%, 55%);
|
||||||
|
--ui-clr-error: hsl(357, 77%, 51%);
|
||||||
|
|
||||||
|
--ui-anim: all 150ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (prefers-color-scheme: light) {
|
||||||
|
:root {
|
||||||
|
--ui-bg-0: #fefefe;
|
||||||
|
--ui-bg-0-85: #fefefed9;
|
||||||
|
--ui-bg-1: #eee;
|
||||||
|
--ui-bg-2: #e2e2e2;
|
||||||
|
--ui-text-0: #111;
|
||||||
|
--ui-text-1: #333;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
:root[theme='dark'] {
|
||||||
|
--ui-bg-0: #111 !important;
|
||||||
|
--ui-bg-0-85: #111111d9 !important;
|
||||||
|
--ui-bg-1: #333 !important;
|
||||||
|
--ui-bg-2: #444 !important;
|
||||||
|
--ui-text-0: #fefefe !important;
|
||||||
|
--ui-text-1: #eee !important;
|
||||||
|
}
|
||||||
|
:root[theme='light'] {
|
||||||
|
--ui-bg-0: #fefefe !important;
|
||||||
|
--ui-bg-0-85: #fefefed9 !important;
|
||||||
|
--ui-bg-1: #eee !important;
|
||||||
|
--ui-bg-2: #e2e2e2 !important;
|
||||||
|
--ui-text-0: #111 !important;
|
||||||
|
--ui-text-1: #333 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.error-text {
|
||||||
|
color: var(--ui-clr-error);
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
min-height: 100vh;
|
||||||
|
margin: 0;
|
||||||
|
background-color: var(--ui-bg-0);
|
||||||
|
color: var(--ui-text-0);
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: inherit;
|
||||||
|
box-sizing: content-box;
|
||||||
|
border-bottom: 2px solid var(--ui-bg-2);
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:active {
|
||||||
|
border-color: var(--ui-clr-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover {
|
||||||
|
border-color: var(--ui-text-0);
|
||||||
|
}
|
||||||
|
|
||||||
|
input,
|
||||||
|
textarea,
|
||||||
|
button {
|
||||||
|
appearance: none;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
border-radius: 0;
|
||||||
|
transition: var(--ui-anim);
|
||||||
|
font-family: inherit;
|
||||||
|
font-size: inherit;
|
||||||
|
background: inherit;
|
||||||
|
color: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
*:disabled,
|
||||||
|
*[disabled='true'] {
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
fieldset {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.box {
|
||||||
|
width: 100%;
|
||||||
|
min-height: min(calc(100vh - 30rem), 20rem);
|
||||||
|
margin: 0;
|
||||||
|
border: 2px solid var(--ui-bg-1);
|
||||||
|
resize: vertical;
|
||||||
|
outline: none;
|
||||||
|
padding: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 30rem) {
|
||||||
|
.box {
|
||||||
|
min-height: calc(100vh - 25rem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.box:hover,
|
||||||
|
.box:focus {
|
||||||
|
border-color: var(--ui-clr-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tr {
|
||||||
|
text-align: right;
|
||||||
|
}
|
@@ -5,9 +5,9 @@
|
|||||||
<link rel="icon" href="/favicon.png" />
|
<link rel="icon" href="/favicon.png" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
|
|
||||||
%svelte.head%
|
%sveltekit.head%
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="svelte">%svelte.body%</div>
|
<div id="svelte">%sveltekit.body%</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
74
frontend/src/lib/api.ts
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
export type NoteMeta = { type: 'text' | 'file' }
|
||||||
|
|
||||||
|
export type Note = {
|
||||||
|
contents: string
|
||||||
|
meta: NoteMeta
|
||||||
|
views?: number
|
||||||
|
expiration?: number
|
||||||
|
}
|
||||||
|
export type NoteInfo = {}
|
||||||
|
export type NotePublic = Pick<Note, 'contents' | 'meta'>
|
||||||
|
export type NoteCreate = Omit<Note, 'meta'> & { meta: string }
|
||||||
|
|
||||||
|
export type FileDTO = Pick<File, 'name' | 'size' | 'type'> & {
|
||||||
|
contents: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type CallOptions = {
|
||||||
|
url: string
|
||||||
|
method: string
|
||||||
|
body?: any
|
||||||
|
}
|
||||||
|
|
||||||
|
export class PayloadToLargeError extends Error {}
|
||||||
|
|
||||||
|
export async function call(options: CallOptions) {
|
||||||
|
const response = await fetch('/api/' + options.url, {
|
||||||
|
method: options.method,
|
||||||
|
body: options.body === undefined ? undefined : JSON.stringify(options.body),
|
||||||
|
mode: 'cors',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
if (response.status === 413) throw new PayloadToLargeError()
|
||||||
|
else throw new Error('API call failed')
|
||||||
|
}
|
||||||
|
return response.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function create(note: Note) {
|
||||||
|
const { meta, ...rest } = note
|
||||||
|
const body: NoteCreate = {
|
||||||
|
...rest,
|
||||||
|
meta: JSON.stringify(meta),
|
||||||
|
}
|
||||||
|
const data = await call({
|
||||||
|
url: 'notes/',
|
||||||
|
method: 'post',
|
||||||
|
body,
|
||||||
|
})
|
||||||
|
return data as { id: string }
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function get(id: string): Promise<NotePublic> {
|
||||||
|
const data = await call({
|
||||||
|
url: `notes/${id}`,
|
||||||
|
method: 'delete',
|
||||||
|
})
|
||||||
|
const { contents, meta } = data
|
||||||
|
return {
|
||||||
|
contents,
|
||||||
|
meta: JSON.parse(meta) as NoteMeta,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function info(id: string): Promise<NoteInfo> {
|
||||||
|
const data = await call({
|
||||||
|
url: `notes/${id}`,
|
||||||
|
method: 'get',
|
||||||
|
})
|
||||||
|
return data
|
||||||
|
}
|
@@ -36,7 +36,7 @@ export function getKeyFromString(password: string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function getDerivedForKey(key: CryptoKey, salt: ArrayBuffer) {
|
export async function getDerivedForKey(key: CryptoKey, salt: ArrayBuffer) {
|
||||||
const iterations = 1_000
|
const iterations = 100_000
|
||||||
return window.crypto.subtle.deriveKey(
|
return window.crypto.subtle.deriveKey(
|
||||||
{
|
{
|
||||||
name: 'PBKDF2',
|
name: 'PBKDF2',
|
13
frontend/src/lib/files.ts
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
export class Files {
|
||||||
|
static toString(f: File | Blob): Promise<string> {
|
||||||
|
const reader = new window.FileReader()
|
||||||
|
reader.readAsDataURL(f)
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
reader.onloadend = () => resolve(reader.result as string)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
static async fromString(s: string): Promise<Blob> {
|
||||||
|
return fetch(s).then((r) => r.blob())
|
||||||
|
}
|
||||||
|
}
|
5
frontend/src/lib/icons/IconContrast.svelte
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="ionicon" viewBox="0 0 512 512"
|
||||||
|
><title>Contrast</title><path
|
||||||
|
d="M256 32C132.29 32 32 132.29 32 256s100.29 224 224 224 224-100.29 224-224S379.71 32 256 32zM128.72 383.28A180 180 0 01256 76v360a178.82 178.82 0 01-127.28-52.72z"
|
||||||
|
/></svg
|
||||||
|
>
|
After Width: | Height: | Size: 287 B |
7
frontend/src/lib/icons/IconCopy.svelte
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="ionicon" viewBox="0 0 512 512"
|
||||||
|
><title>Copy</title><path
|
||||||
|
d="M456 480H136a24 24 0 01-24-24V128a16 16 0 0116-16h328a24 24 0 0124 24v320a24 24 0 01-24 24z"
|
||||||
|
/><path
|
||||||
|
d="M112 80h288V56a24 24 0 00-24-24H60a28 28 0 00-28 28v316a24 24 0 0024 24h24V112a32 32 0 0132-32z"
|
||||||
|
/></svg
|
||||||
|
>
|
After Width: | Height: | Size: 325 B |
5
frontend/src/lib/icons/IconDice.svelte
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="ionicon" viewBox="0 0 512 512"
|
||||||
|
><title>Dice</title><path
|
||||||
|
d="M48 366.92L240 480V284L48 170zM192 288c8.84 0 16 10.75 16 24s-7.16 24-16 24-16-10.75-16-24 7.16-24 16-24zm-96 32c8.84 0 16 10.75 16 24s-7.16 24-16 24-16-10.75-16-24 7.16-24 16-24zM272 284v196l192-113.08V170zm48 140c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm0-88c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm96 32c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm0-88c-8.84 0-16-10.75-16-24s7.16-24 16-24 16 10.75 16 24-7.16 24-16 24zm32 77.64zM256 32L64 144l192 112 192-112zm0 120c-13.25 0-24-7.16-24-16s10.75-16 24-16 24 7.16 24 16-10.75 16-24 16z"
|
||||||
|
/></svg
|
||||||
|
>
|
After Width: | Height: | Size: 736 B |
5
frontend/src/lib/icons/IconEye.svelte
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="ionicon" viewBox="0 0 512 512"
|
||||||
|
><title>Eye</title><circle cx="256" cy="256" r="64" /><path
|
||||||
|
d="M394.82 141.18C351.1 111.2 304.31 96 255.76 96c-43.69 0-86.28 13-126.59 38.48C88.52 160.23 48.67 207 16 256c26.42 44 62.56 89.24 100.2 115.18C159.38 400.92 206.33 416 255.76 416c49 0 95.85-15.07 139.3-44.79C433.31 345 469.71 299.82 496 256c-26.38-43.43-62.9-88.56-101.18-114.82zM256 352a96 96 0 1196-96 96.11 96.11 0 01-96 96z"
|
||||||
|
/></svg
|
||||||
|
>
|
After Width: | Height: | Size: 483 B |
7
frontend/src/lib/icons/IconEyeOff.svelte
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="ionicon" viewBox="0 0 512 512"
|
||||||
|
><title>Eye Off</title><path
|
||||||
|
d="M63.998 86.004l21.998-21.998L448 426.01l-21.998 21.998zM259.34 192.09l60.57 60.57a64.07 64.07 0 00-60.57-60.57zM252.66 319.91l-60.57-60.57a64.07 64.07 0 0060.57 60.57z"
|
||||||
|
/><path
|
||||||
|
d="M256 352a96 96 0 01-92.6-121.34l-69.07-69.08C66.12 187.42 39.24 221.14 16 256c26.42 44 62.56 89.24 100.2 115.18C159.38 400.92 206.33 416 255.76 416A233.47 233.47 0 00335 402.2l-53.61-53.6A95.84 95.84 0 01256 352zM256 160a96 96 0 0192.6 121.34L419.26 352c29.15-26.25 56.07-61.56 76.74-96-26.38-43.43-62.9-88.56-101.18-114.82C351.1 111.2 304.31 96 255.76 96a222.92 222.92 0 00-78.21 14.29l53.11 53.11A95.84 95.84 0 01256 160z"
|
||||||
|
/></svg
|
||||||
|
>
|
After Width: | Height: | Size: 732 B |
20
frontend/src/lib/stores/status.ts
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import { call } from '$lib/api'
|
||||||
|
import { writable } from 'svelte/store'
|
||||||
|
|
||||||
|
export type Status = {
|
||||||
|
version: string
|
||||||
|
max_size: number
|
||||||
|
max_views: number
|
||||||
|
max_expiration: number
|
||||||
|
allow_advanced: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export const status = writable<null | Status>(null)
|
||||||
|
|
||||||
|
export async function init() {
|
||||||
|
const data = await call({
|
||||||
|
url: 'status/',
|
||||||
|
method: 'get',
|
||||||
|
})
|
||||||
|
status.set(data)
|
||||||
|
}
|
19
frontend/src/lib/ui/AboutParagraph.svelte
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
export let title: string
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<b>▶ {title}</b>
|
||||||
|
<slot />
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
b {
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
p > :global(span) {
|
||||||
|
padding-left: 1.25em;
|
||||||
|
}
|
||||||
|
</style>
|
82
frontend/src/lib/ui/FileUpload.svelte
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import type { FileDTO } from '$lib/api'
|
||||||
|
import { Files } from '$lib/files'
|
||||||
|
import { createEventDispatcher } from 'svelte'
|
||||||
|
import { t } from 'svelte-intl-precompile'
|
||||||
|
import Button from './Button.svelte'
|
||||||
|
import MaxSize from './MaxSize.svelte'
|
||||||
|
|
||||||
|
export let label: string = ''
|
||||||
|
let files: File[] = []
|
||||||
|
|
||||||
|
const dispatch = createEventDispatcher<{ file: string }>()
|
||||||
|
|
||||||
|
async function onInput(e: Event) {
|
||||||
|
const input = e.target as HTMLInputElement
|
||||||
|
if (input?.files?.length) {
|
||||||
|
files = [...files, ...Array.from(input.files)]
|
||||||
|
const data: FileDTO[] = await Promise.all(
|
||||||
|
files.map(async (file) => ({
|
||||||
|
name: file.name,
|
||||||
|
type: file.type,
|
||||||
|
size: file.size,
|
||||||
|
contents: await Files.toString(file),
|
||||||
|
}))
|
||||||
|
)
|
||||||
|
dispatch('file', JSON.stringify(data))
|
||||||
|
} else {
|
||||||
|
dispatch('file', '')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function clear(e: Event) {
|
||||||
|
e.preventDefault()
|
||||||
|
files = []
|
||||||
|
dispatch('file', '')
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<label>
|
||||||
|
<small>
|
||||||
|
{label}
|
||||||
|
</small>
|
||||||
|
<input type="file" on:change={onInput} multiple />
|
||||||
|
<div class="box">
|
||||||
|
{#if files.length}
|
||||||
|
<div>
|
||||||
|
<b>{$t('file_upload.selected_files')}</b>
|
||||||
|
{#each files as file}
|
||||||
|
<div class="file">
|
||||||
|
{file.name}
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
<div class="spacer" />
|
||||||
|
<Button on:click={clear}>Clear</Button>
|
||||||
|
</div>
|
||||||
|
{:else}
|
||||||
|
<div>
|
||||||
|
<b>{$t('file_upload.no_files_selected')}</b>
|
||||||
|
<br />
|
||||||
|
<small>{$t('common.max')}: <MaxSize /></small>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
input {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.box {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spacer {
|
||||||
|
margin-top: 1rem;
|
||||||
|
}
|
||||||
|
</style>
|
37
frontend/src/lib/ui/Icon.svelte
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
<script lang="ts" context="module">
|
||||||
|
import IconContrast from '$lib/icons/IconContrast.svelte'
|
||||||
|
import IconCopy from '$lib/icons/IconCopy.svelte'
|
||||||
|
import IconDice from '$lib/icons/IconDice.svelte'
|
||||||
|
import IconEye from '$lib/icons/IconEye.svelte'
|
||||||
|
import IconEyeOff from '$lib/icons/IconEyeOff.svelte'
|
||||||
|
|
||||||
|
const map = {
|
||||||
|
contrast: IconContrast,
|
||||||
|
copy: IconCopy,
|
||||||
|
dice: IconDice,
|
||||||
|
eye: IconEye,
|
||||||
|
'eye-off': IconEyeOff,
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
export let icon: keyof typeof map
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div on:click {...$$restProps}>
|
||||||
|
{#if map[icon]}
|
||||||
|
<svelte:component this={map[icon]} />
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
div {
|
||||||
|
display: inline-block;
|
||||||
|
contain: strict;
|
||||||
|
box-sizing: content-box;
|
||||||
|
}
|
||||||
|
div > :global(svg) {
|
||||||
|
display: block;
|
||||||
|
fill: currentColor;
|
||||||
|
}
|
||||||
|
</style>
|
13
frontend/src/lib/ui/MaxSize.svelte
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { status } from '$lib/stores/status'
|
||||||
|
import prettyBytes from 'pretty-bytes'
|
||||||
|
import { _ } from 'svelte-intl-precompile'
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<span>
|
||||||
|
{#if $status !== null}
|
||||||
|
{prettyBytes($status.max_size, { binary: true })}
|
||||||
|
{:else}
|
||||||
|
{$_('common.loading')}
|
||||||
|
{/if}
|
||||||
|
</span>
|
87
frontend/src/lib/ui/ShowNote.svelte
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import type { FileDTO, NotePublic } from '$lib/api'
|
||||||
|
import { Files } from '$lib/files'
|
||||||
|
import copy from 'copy-to-clipboard'
|
||||||
|
import { saveAs } from 'file-saver'
|
||||||
|
import prettyBytes from 'pretty-bytes'
|
||||||
|
import sanitize from 'sanitize-html'
|
||||||
|
import { t } from 'svelte-intl-precompile'
|
||||||
|
import Button from './Button.svelte'
|
||||||
|
|
||||||
|
export let note: NotePublic
|
||||||
|
|
||||||
|
const RE_URL = /[A-Za-z]+:\/\/([A-Z a-z0-9\-._~:\/?#\[\]@!$&'()*+,;%=])+/g
|
||||||
|
let files: FileDTO[] = []
|
||||||
|
|
||||||
|
$: if (note.meta.type === 'file') {
|
||||||
|
files = JSON.parse(note.contents) as FileDTO[]
|
||||||
|
}
|
||||||
|
|
||||||
|
$: download = () => {
|
||||||
|
for (const file of files) {
|
||||||
|
downloadFile(file)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function downloadFile(file: FileDTO) {
|
||||||
|
const f = new File([await Files.fromString(file.contents)], file.name, {
|
||||||
|
type: file.type,
|
||||||
|
})
|
||||||
|
saveAs(f)
|
||||||
|
}
|
||||||
|
|
||||||
|
function contentWithLinks(content: string): string {
|
||||||
|
const replaced = note.contents.replace(
|
||||||
|
RE_URL,
|
||||||
|
(url) => `<a href="${url}" rel="noreferrer">${url}</a>`
|
||||||
|
)
|
||||||
|
return sanitize(replaced, { allowedTags: ['a'], allowedAttributes: { a: ['href', 'rel'] } })
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<p class="error-text">{@html $t('show.warning_will_not_see_again')}</p>
|
||||||
|
{#if note.meta.type === 'text'}
|
||||||
|
<div class="note">
|
||||||
|
{@html contentWithLinks(note.contents)}
|
||||||
|
</div>
|
||||||
|
<Button on:click={() => copy(note.contents)}>{$t('common.copy_clipboard')}</Button>
|
||||||
|
{:else}
|
||||||
|
{#each files as file}
|
||||||
|
<div class="note file">
|
||||||
|
<b on:click={() => downloadFile(file)}>↓ {file.name}</b>
|
||||||
|
<small> {file.type} - {prettyBytes(file.size)}</small>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
<Button on:click={download}>{$t('show.download_all')}</Button>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.note {
|
||||||
|
width: 100%;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
border: 2px solid var(--ui-bg-1);
|
||||||
|
outline: none;
|
||||||
|
padding: 0.5rem;
|
||||||
|
white-space: pre;
|
||||||
|
overflow: auto;
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.note b {
|
||||||
|
cursor: pointer;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
.note.file {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.note.file small {
|
||||||
|
padding-left: 1rem;
|
||||||
|
}
|
||||||
|
</style>
|
@@ -4,7 +4,7 @@
|
|||||||
export let color = true
|
export let color = true
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div>
|
<div {...$$restProps}>
|
||||||
<label class="switch">
|
<label class="switch">
|
||||||
<small>{label}</small>
|
<small>{label}</small>
|
||||||
<input type="checkbox" bind:checked={value} />
|
<input type="checkbox" bind:checked={value} />
|
||||||
@@ -49,7 +49,7 @@
|
|||||||
height: 2rem;
|
height: 2rem;
|
||||||
width: 1.25rem;
|
width: 1.25rem;
|
||||||
left: 0.125rem;
|
left: 0.125rem;
|
||||||
bottom: 0.1rem;
|
bottom: 0.125rem;
|
||||||
background-color: var(--ui-bg-1);
|
background-color: var(--ui-bg-1);
|
||||||
-webkit-transition: 0.4s;
|
-webkit-transition: 0.4s;
|
||||||
transition: var(--ui-anim);
|
transition: var(--ui-anim);
|
11
frontend/src/lib/ui/TextArea.svelte
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
export let label: string = ''
|
||||||
|
export let value: string
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<label>
|
||||||
|
<small>
|
||||||
|
{label}
|
||||||
|
</small>
|
||||||
|
<textarea class="box" {...$$restProps} bind:value />
|
||||||
|
</label>
|
@@ -1,19 +1,23 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { getRandomBytes, Hex } from '$lib/crypto'
|
import { getRandomBytes, Hex } from '$lib/crypto'
|
||||||
|
|
||||||
import copyToClipboard from 'copy-to-clipboard'
|
import copyToClipboard from 'copy-to-clipboard'
|
||||||
|
import { t } from 'svelte-intl-precompile'
|
||||||
|
import { fade } from 'svelte/transition'
|
||||||
import Icon from './Icon.svelte'
|
import Icon from './Icon.svelte'
|
||||||
|
|
||||||
export let label: string = ''
|
export let label: string = ''
|
||||||
export let value
|
export let value: any
|
||||||
|
export let validate: (value: any) => boolean | string = () => true
|
||||||
export let copy: boolean = false
|
export let copy: boolean = false
|
||||||
export let random: boolean = false
|
export let random: boolean = false
|
||||||
|
|
||||||
const initialType = $$restProps.type
|
const initialType = $$restProps.type
|
||||||
const isPassword = initialType === 'password'
|
const isPassword = initialType === 'password'
|
||||||
let hidden = true
|
let hidden = true
|
||||||
|
let notification: string | null = null
|
||||||
|
let notificationTimeout: NodeJS.Timeout | null = null
|
||||||
|
|
||||||
|
$: valid = validate(value)
|
||||||
|
|
||||||
$: if (isPassword) {
|
$: if (isPassword) {
|
||||||
value
|
value
|
||||||
@@ -24,29 +28,46 @@
|
|||||||
hidden = !hidden
|
hidden = !hidden
|
||||||
}
|
}
|
||||||
function copyFN() {
|
function copyFN() {
|
||||||
copyToClipboard(value)
|
copyToClipboard(value.toString())
|
||||||
|
notify($t('home.copied_to_clipboard'))
|
||||||
}
|
}
|
||||||
function randomFN() {
|
function randomFN() {
|
||||||
value = Hex.encode(getRandomBytes(20))
|
value = Hex.encode(getRandomBytes(20))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function notify(msg: string, delay: number = 2000) {
|
||||||
|
if (notificationTimeout) {
|
||||||
|
clearTimeout(notificationTimeout)
|
||||||
|
}
|
||||||
|
notificationTimeout = setTimeout(() => {
|
||||||
|
notification = null
|
||||||
|
}, delay)
|
||||||
|
notification = msg
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<label>
|
<label>
|
||||||
<small disabled={$$restProps.disabled}>
|
<small disabled={$$restProps.disabled}>
|
||||||
{label}
|
{label}
|
||||||
|
{#if valid !== true}
|
||||||
|
<span class="error-text">{valid}</span>
|
||||||
|
{/if}
|
||||||
</small>
|
</small>
|
||||||
<input bind:value {...$$restProps} />
|
<input bind:value {...$$restProps} class:valid={valid === true} />
|
||||||
<div class="icons">
|
<div class="icons">
|
||||||
{#if isPassword}
|
{#if isPassword}
|
||||||
<Icon class="icon" icon={hidden ? 'eye-sharp' : 'eye-off-sharp'} on:click={toggle} />
|
<Icon class="icon" icon={hidden ? 'eye' : 'eye-off'} on:click={toggle} />
|
||||||
{/if}
|
{/if}
|
||||||
{#if random}
|
{#if random}
|
||||||
<Icon class="icon" icon="dice-sharp" on:click={randomFN} />
|
<Icon class="icon" icon="dice" on:click={randomFN} />
|
||||||
{/if}
|
{/if}
|
||||||
{#if copy}
|
{#if copy}
|
||||||
<Icon class="icon" icon="copy-sharp" on:click={copyFN} />
|
<Icon class="icon" icon="copy" on:click={copyFN} />
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
|
{#if notification}
|
||||||
|
<div class="notification" transition:fade><small>{notification}</small></div>
|
||||||
|
{/if}
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
@@ -55,6 +76,10 @@
|
|||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
label > small {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
input {
|
input {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
@@ -68,6 +93,10 @@
|
|||||||
border-color: var(--ui-clr-primary);
|
border-color: var(--ui-clr-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
input:not(.valid) {
|
||||||
|
border-color: var(--ui-clr-error);
|
||||||
|
}
|
||||||
|
|
||||||
.icons {
|
.icons {
|
||||||
border: 1px red;
|
border: 1px red;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@@ -88,4 +117,11 @@
|
|||||||
.icons > :global(.icon:hover) {
|
.icons > :global(.icon:hover) {
|
||||||
border-color: var(--ui-clr-primary);
|
border-color: var(--ui-clr-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.notification {
|
||||||
|
text-align: right;
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
bottom: -1.5em;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
62
frontend/src/lib/ui/ThemeToggle.svelte
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
<script lang="ts" context="module">
|
||||||
|
import { writable } from 'svelte/store'
|
||||||
|
|
||||||
|
enum Theme {
|
||||||
|
Dark = 'dark',
|
||||||
|
Light = 'light',
|
||||||
|
Auto = 'auto',
|
||||||
|
}
|
||||||
|
|
||||||
|
const NextTheme = {
|
||||||
|
[Theme.Auto]: Theme.Light,
|
||||||
|
[Theme.Light]: Theme.Dark,
|
||||||
|
[Theme.Dark]: Theme.Auto,
|
||||||
|
}
|
||||||
|
|
||||||
|
function init(): Theme {
|
||||||
|
if (typeof window !== 'undefined') {
|
||||||
|
const saved = window.localStorage.getItem('theme') as Theme
|
||||||
|
if (Object.values(Theme).includes(saved)) return saved
|
||||||
|
}
|
||||||
|
return Theme.Auto
|
||||||
|
}
|
||||||
|
|
||||||
|
export const theme = writable<Theme>(init())
|
||||||
|
|
||||||
|
theme.subscribe((theme) => {
|
||||||
|
if (typeof window !== 'undefined') {
|
||||||
|
window.localStorage.setItem('theme', theme)
|
||||||
|
const html = window.document.getElementsByTagName('html')[0]
|
||||||
|
html.setAttribute('theme', theme)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import Icon from '$lib/ui/Icon.svelte'
|
||||||
|
|
||||||
|
function change() {
|
||||||
|
theme.update((current) => NextTheme[current])
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div on:click={change}>
|
||||||
|
<Icon class="icon" icon="contrast" />
|
||||||
|
{$theme}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
div :global(.icon) {
|
||||||
|
height: 1rem;
|
||||||
|
width: 1rem;
|
||||||
|
margin-right: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
div {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: flex-end;
|
||||||
|
align-items: center;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
</style>
|
200
frontend/src/lib/views/Create.svelte
Normal file
@@ -0,0 +1,200 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { create, Note, PayloadToLargeError } from '$lib/api'
|
||||||
|
import { encrypt, getKeyFromString, getRandomBytes, Hex } from '$lib/crypto'
|
||||||
|
import { status } from '$lib/stores/status'
|
||||||
|
import Button from '$lib/ui/Button.svelte'
|
||||||
|
import FileUpload from '$lib/ui/FileUpload.svelte'
|
||||||
|
import MaxSize from '$lib/ui/MaxSize.svelte'
|
||||||
|
import Switch from '$lib/ui/Switch.svelte'
|
||||||
|
import TextArea from '$lib/ui/TextArea.svelte'
|
||||||
|
import TextInput from '$lib/ui/TextInput.svelte'
|
||||||
|
import { t } from 'svelte-intl-precompile'
|
||||||
|
import { blur } from 'svelte/transition'
|
||||||
|
|
||||||
|
let note: Note = {
|
||||||
|
contents: '',
|
||||||
|
meta: { type: 'text' },
|
||||||
|
views: 1,
|
||||||
|
expiration: 60,
|
||||||
|
}
|
||||||
|
let result: { password: string; id: string } | null = null
|
||||||
|
let advanced = false
|
||||||
|
let file = false
|
||||||
|
let timeExpiration = false
|
||||||
|
let message = ''
|
||||||
|
let loading = false
|
||||||
|
let error: string | null = null
|
||||||
|
|
||||||
|
$: if (!advanced) {
|
||||||
|
note.views = 1
|
||||||
|
timeExpiration = false
|
||||||
|
}
|
||||||
|
|
||||||
|
$: {
|
||||||
|
message = $t('home.explanation', {
|
||||||
|
values: {
|
||||||
|
type: $t(timeExpiration ? 'common.minutes' : 'common.views', {
|
||||||
|
values: { n: (timeExpiration ? note.expiration : note.views) ?? '?' },
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
$: note.meta.type = file ? 'file' : 'text'
|
||||||
|
|
||||||
|
$: if (!file) {
|
||||||
|
note.contents = ''
|
||||||
|
}
|
||||||
|
|
||||||
|
class EmptyContentError extends Error {}
|
||||||
|
|
||||||
|
async function submit() {
|
||||||
|
try {
|
||||||
|
error = null
|
||||||
|
loading = true
|
||||||
|
const password = Hex.encode(getRandomBytes(32))
|
||||||
|
const key = await getKeyFromString(password)
|
||||||
|
if (note.contents === '') throw new EmptyContentError()
|
||||||
|
const data: Note = {
|
||||||
|
contents: await encrypt(note.contents, key),
|
||||||
|
meta: note.meta,
|
||||||
|
}
|
||||||
|
if (timeExpiration) data.expiration = parseInt(note.expiration as any)
|
||||||
|
else data.views = parseInt(note.views as any)
|
||||||
|
|
||||||
|
const response = await create(data)
|
||||||
|
result = {
|
||||||
|
password: password,
|
||||||
|
id: response.id,
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
if (e instanceof PayloadToLargeError) {
|
||||||
|
error = $t('home.errors.note_to_big')
|
||||||
|
} else if (e instanceof EmptyContentError) {
|
||||||
|
error = $t('home.errors.empty_content')
|
||||||
|
} else {
|
||||||
|
error = $t('home.errors.note_error')
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
loading = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function reset() {
|
||||||
|
window.location.reload()
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{#if result}
|
||||||
|
<TextInput
|
||||||
|
type="text"
|
||||||
|
readonly
|
||||||
|
label={$t('common.share_link')}
|
||||||
|
value="{window.location.origin}/note/{result.id}#{result.password}"
|
||||||
|
copy
|
||||||
|
/>
|
||||||
|
<br />
|
||||||
|
<p>
|
||||||
|
{@html $t('home.new_note_notice')}
|
||||||
|
</p>
|
||||||
|
<br />
|
||||||
|
<Button on:click={reset}>{$t('home.new_note')}</Button>
|
||||||
|
{:else}
|
||||||
|
<p>
|
||||||
|
{@html $t('home.intro')}
|
||||||
|
</p>
|
||||||
|
<form on:submit|preventDefault={submit}>
|
||||||
|
<fieldset disabled={loading}>
|
||||||
|
{#if file}
|
||||||
|
<FileUpload label={$t('common.file')} on:file={(f) => (note.contents = f.detail)} />
|
||||||
|
{:else}
|
||||||
|
<TextArea label={$t('common.note')} bind:value={note.contents} placeholder="..." />
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<div class="bottom">
|
||||||
|
<Switch class="file" label={$t('common.file')} bind:value={file} />
|
||||||
|
{#if $status?.allow_advanced}
|
||||||
|
<Switch label={$t('common.advanced')} bind:value={advanced} />
|
||||||
|
{/if}
|
||||||
|
<div class="grow" />
|
||||||
|
<div class="tr">
|
||||||
|
<small>{$t('common.max')}: <MaxSize /> </small>
|
||||||
|
<br />
|
||||||
|
<Button type="submit">{$t('common.create')}</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{#if error}
|
||||||
|
<div class="error-text">{error}</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<br />
|
||||||
|
{#if loading}
|
||||||
|
{$t('common.loading')}
|
||||||
|
{:else}
|
||||||
|
{message}
|
||||||
|
{/if}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{#if advanced}
|
||||||
|
<div transition:blur={{ duration: 250 }}>
|
||||||
|
<br />
|
||||||
|
<div class="fields">
|
||||||
|
<TextInput
|
||||||
|
type="number"
|
||||||
|
label={$t('common.views', { values: { n: 0 } })}
|
||||||
|
bind:value={note.views}
|
||||||
|
disabled={timeExpiration}
|
||||||
|
max={$status?.max_views}
|
||||||
|
validate={(v) =>
|
||||||
|
($status && v < $status?.max_views) ||
|
||||||
|
$t('home.errors.max', { values: { n: $status?.max_views ?? 0 } })}
|
||||||
|
/>
|
||||||
|
<div class="middle-switch">
|
||||||
|
<Switch label={$t('common.mode')} bind:value={timeExpiration} color={false} />
|
||||||
|
</div>
|
||||||
|
<TextInput
|
||||||
|
type="number"
|
||||||
|
label={$t('common.minutes', { values: { n: 0 } })}
|
||||||
|
bind:value={note.expiration}
|
||||||
|
disabled={!timeExpiration}
|
||||||
|
max={$status?.max_expiration}
|
||||||
|
validate={(v) =>
|
||||||
|
($status && v < $status?.max_expiration) ||
|
||||||
|
$t('home.errors.max', { values: { n: $status?.max_expiration ?? 0 } })}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</fieldset>
|
||||||
|
</form>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.bottom {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-end;
|
||||||
|
margin-top: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottom :global(.file) {
|
||||||
|
margin-right: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grow {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.middle-switch {
|
||||||
|
margin: 0 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.error-text {
|
||||||
|
margin-top: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fields {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
</style>
|
36
frontend/src/lib/views/Footer.svelte
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import ThemeToggle from '$lib/ui/ThemeToggle.svelte'
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<footer>
|
||||||
|
<ThemeToggle />
|
||||||
|
<nav>
|
||||||
|
<a href="/">/home</a>
|
||||||
|
<a href="/about">/about</a>
|
||||||
|
<a href="https://github.com/cupcakearmy/cryptgeon" target="_blank" rel="noopener">/code</a>
|
||||||
|
</nav>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
footer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 1rem;
|
||||||
|
position: fixed;
|
||||||
|
bottom: 0;
|
||||||
|
right: 0;
|
||||||
|
width: 100%;
|
||||||
|
background-color: var(--ui-bg-0-85);
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
margin: 0 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: flex-end;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
</style>
|
107
frontend/src/lib/views/Header.svelte
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
<header>
|
||||||
|
<a href="/">
|
||||||
|
<svg
|
||||||
|
width="100%"
|
||||||
|
height="100%"
|
||||||
|
viewBox="0 0 450 200"
|
||||||
|
version="1.1"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||||
|
xml:space="preserve"
|
||||||
|
style="fill-rule:evenodd;clip-rule:evenodd;stroke-linejoin:round;stroke-miterlimit:2;"
|
||||||
|
><g id="Logo"
|
||||||
|
><clipPath id="_clip1"><rect x="6.336" y="3.225" width="193.55" height="193.55" /></clipPath
|
||||||
|
><g clip-path="url(#_clip1)"
|
||||||
|
><g
|
||||||
|
><g
|
||||||
|
><path
|
||||||
|
d="M173.425,43.296c-2.087,-0 -3.78,1.693 -3.78,3.78c-0,2.087 1.693,3.78 3.78,3.78c2.087,0 3.78,-1.693 3.78,-3.78c0,-2.087 -1.693,-3.78 -3.78,-3.78Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/></g
|
||||||
|
></g
|
||||||
|
><g
|
||||||
|
><g
|
||||||
|
><path
|
||||||
|
d="M103.112,134.023c-2.087,-0 -3.781,1.693 -3.781,3.78c0,2.087 1.694,3.78 3.781,3.78c2.086,0 3.78,-1.693 3.78,-3.78c-0,-2.087 -1.694,-3.78 -3.78,-3.78Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/></g
|
||||||
|
></g
|
||||||
|
><g
|
||||||
|
><g
|
||||||
|
><path
|
||||||
|
d="M156.036,96.22c-2.088,-0 -3.781,1.692 -3.781,3.78c0,18.76 -15.262,34.023 -34.022,34.023c-2.088,-0 -3.781,1.692 -3.781,3.78c0,2.088 1.693,3.78 3.781,3.78c22.929,0 41.583,-18.654 41.583,-41.583c-0,-2.088 -1.693,-3.78 -3.78,-3.78Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/></g
|
||||||
|
></g
|
||||||
|
><g
|
||||||
|
><g
|
||||||
|
><path
|
||||||
|
d="M199.488,60.507l-9.515,-19.026c-4.102,-8.207 -12.35,-13.306 -21.527,-13.306c-7.479,-0 -14.626,3.547 -19.154,9.498l-10.679,12.016c-2.846,-4.047 -7.021,-7.049 -11.83,-8.421l-19.102,-5.459c-14.623,-4.178 -28.92,-15.441 -39.227,-30.901c-0.924,-1.386 -2.646,-2.003 -4.241,-1.521c-1.594,0.483 -2.684,1.953 -2.684,3.618l-0,20.372c-0,9.468 1.417,18.804 4.219,27.813c-2.936,0.73 -5.896,1.34 -8.843,1.816c-5.772,0.936 -11.653,1.411 -17.48,1.411l-29.308,-0c-1.374,-0 -2.64,0.746 -3.307,1.948c-0.666,1.202 -0.627,2.671 0.101,3.836l22.637,36.219c9.672,15.473 26.329,25.183 44.578,25.983l-36.36,41.158c-5.602,5.728 -3.655,15.315 3.746,18.396l20.017,9.887c0.089,0.044 0.179,0.084 0.271,0.121c5.79,2.313 12.389,-0.496 14.726,-6.279l13.969,-32.982l27.738,0c31.966,0 58.972,-25.967 58.972,-56.704l0,-22.682c0,-6.253 5.088,-11.341 11.341,-11.341l7.56,0c1.311,0 2.528,-0.678 3.216,-1.793c0.689,-1.114 0.752,-2.506 0.166,-3.677Zm-130.399,-42.236c10.418,12.203 23.307,21.034 36.515,24.808l19.103,5.459c3.726,1.063 6.866,3.624 8.666,7.048l-10.048,11.307l-17.652,-10.591c-7.646,-4.588 -16.746,-6.412 -25.776,-4.951c-2.838,0.46 -4.649,1.038 -6.877,1.759c-2.609,-8.332 -3.931,-16.971 -3.931,-25.733l0,-9.106Zm119.457,40.146c-10.422,-0 -18.901,8.479 -18.901,18.901l-0,22.682c-0,26.639 -23.544,49.144 -51.412,49.144l-30.243,-0c-10.77,-0 -20.451,5.983 -25.265,15.615l-0.797,1.596c-0.934,1.867 -0.177,4.137 1.691,5.071c1.867,0.934 4.138,0.176 5.071,-1.691c0.44,-0.586 3.306,-9.102 13.21,-12.125l-12.349,29.159c-0.01,0.024 -0.02,0.047 -0.029,0.071c-0.75,1.877 -2.864,2.851 -4.8,2.148c-21.279,-10.506 -19.997,-9.888 -20.252,-9.99c-2.526,-1.01 -3.191,-4.259 -1.267,-6.182c0.13,-0.131 8.026,-9.078 41.009,-46.411c13.867,-0.617 26.841,-6.319 36.694,-16.172c1.476,-1.476 1.476,-3.87 0,-5.346c-1.476,-1.477 -3.87,-1.476 -5.346,-0c-16.827,16.828 -36.803,13.634 -39.028,14.014c-16.603,0 -31.771,-8.407 -40.573,-22.488l-2.417,-3.868l2.729,1.065c17.307,6.753 38.919,4.347 53.816,-5.586c1.737,-1.158 2.207,-3.505 1.049,-5.242c-1.159,-1.737 -3.505,-2.207 -5.243,-1.048c-13.085,8.724 -31.922,10.666 -46.875,4.832l-12.185,-4.753l-9.896,-15.836l22.488,0c6.231,0 12.519,-0.507 18.689,-1.507c12.711,-2.055 18.051,-4.855 22.992,-5.655c7.182,-1.163 14.516,0.274 20.678,3.97l29.625,17.775c1.789,1.074 4.112,0.494 5.186,-1.296c1.075,-1.79 0.495,-4.113 -1.295,-5.187l-5.378,-3.226c26.56,-29.893 25.14,-28.272 25.32,-28.511c3.101,-4.136 8.038,-6.605 13.204,-6.605c6.294,0 11.951,3.497 14.764,9.127l6.779,13.555l-1.443,-0Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/></g
|
||||||
|
></g
|
||||||
|
></g
|
||||||
|
><g
|
||||||
|
><path
|
||||||
|
d="M222.664,114.692c0.85,-0 1.274,0.586 1.274,1.759c0,1.174 -0.242,2.347 -0.728,3.52c-0.485,1.173 -1.183,2.326 -2.093,3.459c-0.91,1.132 -2.185,2.063 -3.823,2.791c-1.639,0.728 -3.489,1.092 -5.552,1.092c-4.531,0 -8.081,-1.143 -10.65,-3.428c-2.569,-2.286 -3.853,-5.644 -3.853,-10.073c-0,-4.43 1.133,-7.889 3.398,-10.377c2.265,-2.488 5.158,-3.732 8.677,-3.732c2.549,0 4.41,0.678 5.583,2.033c1.173,1.355 1.76,2.964 1.76,4.824c-0,1.012 -0.304,1.821 -0.91,2.428c-0.607,0.606 -1.194,0.91 -1.76,0.91c-1.052,-0 -1.74,-0.243 -2.063,-0.728c0.242,-0.607 0.364,-1.477 0.364,-2.61c-0,-1.132 -0.385,-2.164 -1.153,-3.094c-0.769,-0.931 -1.699,-1.396 -2.792,-1.396c-1.901,0 -3.266,0.93 -4.095,2.791c-0.83,1.861 -1.244,4.784 -1.244,8.769c-0,3.984 0.778,6.867 2.336,8.647c1.557,1.78 3.732,2.67 6.523,2.67c2.791,-0 5.138,-0.86 7.039,-2.579c1.901,-1.72 2.852,-4.157 2.852,-7.312c0.04,-0.243 0.344,-0.364 0.91,-0.364Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M227.337,116.815l-0,9.649c-0.405,0.566 -1.113,0.849 -2.124,0.849c-1.012,0 -1.821,-0.313 -2.428,-0.94c-0.606,-0.627 -0.91,-1.548 -0.91,-2.761l0,-23.059c0.405,-0.566 1.113,-0.85 2.124,-0.85c1.011,0 1.821,0.314 2.427,0.941c0.607,0.627 0.911,1.547 0.911,2.761l-0,4.612c2.022,-5.017 4.814,-7.525 8.373,-7.525c2.549,0 4.41,0.678 5.583,2.033c1.173,1.355 1.76,2.963 1.76,4.824c-0,1.012 -0.304,1.821 -0.91,2.427c-0.607,0.607 -1.194,0.911 -1.76,0.911c-1.052,-0 -1.74,-0.243 -2.063,-0.728c0.242,-0.607 0.364,-1.477 0.364,-2.61c-0,-1.132 -0.384,-2.164 -1.153,-3.094c-0.769,-0.931 -1.699,-1.396 -2.791,-1.396c-1.902,-0 -3.611,1.729 -5.128,5.188c-1.517,3.459 -2.275,6.382 -2.275,8.768Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M245.419,103.405c0,-1.214 0.304,-2.134 0.911,-2.761c0.606,-0.627 1.375,-0.941 2.306,-0.941c0.93,0 1.678,0.284 2.245,0.85l-0,17.415c-0,4.814 1.213,7.221 3.641,7.221c2.184,0 3.974,-1.153 5.37,-3.458c1.395,-2.306 2.093,-6.19 2.093,-11.651l0,-6.675c0,-1.214 0.304,-2.134 0.91,-2.761c0.607,-0.627 1.376,-0.941 2.306,-0.941c0.931,0 1.679,0.284 2.246,0.85l-0,30.826c0.485,-0.041 1.254,-0.061 2.305,-0.061c1.052,0 1.578,0.445 1.578,1.335l-0.364,0.91c-1.537,0 -2.71,0.021 -3.519,0.061l-0,3.034c-0,4.167 -0.941,7.454 -2.822,9.861c-1.881,2.407 -4.399,3.61 -7.555,3.61c-1.861,0 -3.509,-0.556 -4.945,-1.669c-1.436,-1.112 -2.155,-2.862 -2.155,-5.248c0,-3.075 1.093,-5.522 3.277,-7.343c2.185,-1.82 5.097,-3.095 8.738,-3.823l0,-9.466c-1.699,3.156 -4.429,4.733 -8.192,4.733c-2.589,0 -4.632,-0.829 -6.128,-2.488c-1.497,-1.658 -2.246,-4.045 -2.246,-7.16l0,-14.26Zm7.646,39.989c0,1.416 0.385,2.498 1.153,3.246c0.769,0.748 1.699,1.123 2.791,1.123c1.093,-0 1.983,-0.324 2.67,-0.971c1.538,-1.376 2.306,-5.522 2.306,-12.44c-5.946,1.335 -8.92,4.349 -8.92,9.042Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M298.576,104.861c1.335,2.306 2.002,5.411 2.002,9.315c0,3.904 -0.981,7.059 -2.943,9.466c-1.962,2.407 -4.52,3.611 -7.676,3.611c-1.861,-0 -3.509,-0.648 -4.945,-1.942c-1.436,-1.295 -2.155,-3.358 -2.155,-6.19c0,-6.634 3.297,-11.59 9.891,-14.866c-1.051,-1.457 -2.518,-2.185 -4.399,-2.185c-1.881,0 -3.671,0.93 -5.37,2.791c-1.699,1.861 -2.549,4.653 -2.549,8.374l0,36.105c-0.404,0.567 -1.112,0.85 -2.124,0.85c-1.011,-0 -1.82,-0.314 -2.427,-0.941c-0.607,-0.627 -0.91,-1.547 -0.91,-2.761l-0,-45.935c0.405,-0.566 1.112,-0.85 2.124,-0.85c1.011,0 1.82,0.314 2.427,0.941c0.607,0.627 0.91,1.547 0.91,2.761l0,1.335c2.104,-3.358 4.885,-5.037 8.344,-5.037c3.459,0 6.159,0.971 8.101,2.913c4.247,-1.375 9.001,-2.063 14.26,-2.063c1.092,-0 1.638,0.445 1.638,1.335l-0.364,0.91c-4.895,0 -9.507,0.688 -13.835,2.063Zm-8.738,20.025c3.317,-0 4.976,-3.297 4.976,-9.891c-0,-3.479 -0.284,-6.21 -0.85,-8.192c-5.34,2.913 -8.01,7.08 -8.01,12.5c0,1.78 0.385,3.156 1.153,4.127c0.769,0.971 1.679,1.456 2.731,1.456Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M308.285,94.424c-0,-1.213 0.303,-2.134 0.91,-2.761c0.607,-0.627 1.375,-0.94 2.306,-0.94c0.93,-0 1.679,0.283 2.245,0.849l0,8.981l8.313,-0c0.85,-0 1.406,0.162 1.669,0.485c0.263,0.324 0.394,0.87 0.394,1.639l-10.376,-0l0,15.291c0,2.225 0.637,3.945 1.911,5.158c1.275,1.214 2.741,1.821 4.4,1.821c2.508,-0 4.349,-0.931 5.522,-2.792c1.173,-1.861 1.76,-4.874 1.76,-9.041c0.08,-0.243 0.384,-0.364 0.91,-0.364c0.728,-0 1.092,1.072 1.092,3.216c-0,3.277 -0.819,5.987 -2.458,8.131c-1.638,2.144 -4.156,3.216 -7.554,3.216c-3.398,0 -6.089,-0.89 -8.071,-2.67c-1.982,-1.78 -2.973,-4.429 -2.973,-7.949l-0,-22.27Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M345.543,101.463c0.526,-1.173 1.365,-1.76 2.518,-1.76c1.153,0 2.013,0.284 2.579,0.85l-0,30.826c0.485,-0.041 1.254,-0.061 2.306,-0.061c1.052,0 1.578,0.445 1.578,1.335l-0.365,0.91c-1.537,0 -2.71,0.021 -3.519,0.061l-0,3.155c-0.041,4.167 -1.001,7.434 -2.882,9.8c-1.882,2.367 -4.38,3.55 -7.494,3.55c-1.861,0 -3.51,-0.556 -4.946,-1.669c-1.436,-1.112 -2.154,-2.862 -2.154,-5.248c-0,-3.075 1.092,-5.522 3.277,-7.343c2.184,-1.82 5.097,-3.095 8.738,-3.823l-0,-8.92c-1.699,2.792 -4.046,4.187 -7.039,4.187c-7.08,0 -10.619,-4.632 -10.619,-13.896c-0,-4.692 1.102,-8.151 3.307,-10.376c2.204,-2.225 5.127,-3.338 8.768,-3.338c2.791,0 4.774,0.587 5.947,1.76Zm-0.364,12.986l-0,-4.308c-0,-2.873 -0.486,-4.936 -1.457,-6.19c-0.971,-1.254 -2.093,-1.881 -3.367,-1.881c-1.275,0 -2.276,0.121 -3.004,0.364c-0.728,0.243 -1.436,0.748 -2.124,1.517c-1.295,1.456 -1.942,4.612 -1.942,9.466c0,3.196 0.283,5.684 0.85,7.464c0.566,1.78 1.213,2.903 1.942,3.368c0.728,0.465 1.749,0.698 3.064,0.698c1.315,-0 2.65,-0.84 4.005,-2.519c1.355,-1.678 2.033,-4.338 2.033,-7.979Zm-0,22.209l-0,-2.306c-5.947,1.335 -8.92,4.349 -8.92,9.042c-0,1.416 0.384,2.498 1.153,3.246c0.768,0.748 1.678,1.123 2.73,1.123c1.699,-0 2.963,-0.789 3.793,-2.367c0.829,-1.578 1.244,-4.49 1.244,-8.738Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M381.891,113.478c0.849,-0 1.274,0.647 1.274,1.942c-0,2.993 -1.042,5.724 -3.125,8.192c-2.084,2.467 -5.249,3.701 -9.497,3.701c-4.247,0 -7.656,-1.143 -10.224,-3.428c-2.569,-2.286 -3.854,-5.644 -3.854,-10.073c0,-4.43 1.133,-7.889 3.398,-10.377c2.266,-2.488 5.158,-3.732 8.678,-3.732c2.548,0 4.409,0.678 5.582,2.033c1.174,1.355 1.76,2.964 1.76,4.824c0,3.034 -1.264,5.553 -3.792,7.555c-2.529,2.003 -5.755,3.004 -9.679,3.004c0.364,2.589 1.305,4.541 2.822,5.856c1.517,1.314 3.519,1.972 6.007,1.972c2.488,-0 4.733,-0.941 6.736,-2.822c2.002,-1.881 3.003,-4.642 3.003,-8.283c0.041,-0.243 0.344,-0.364 0.911,-0.364Zm-19.661,0.85l-0,0.546c2.508,-0.324 4.682,-1.113 6.523,-2.367c1.841,-1.254 2.761,-3.115 2.761,-5.583c0,-1.577 -0.344,-2.781 -1.031,-3.61c-0.688,-0.829 -1.659,-1.244 -2.913,-1.244c-1.901,0 -3.267,0.93 -4.096,2.791c-0.829,1.861 -1.244,5.017 -1.244,9.467Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M393.359,99.703c3.601,0 6.534,1.194 8.799,3.581c2.265,2.386 3.398,5.633 3.398,9.739c0,4.106 -1.062,7.514 -3.186,10.225c-2.124,2.71 -4.965,4.065 -8.525,4.065c-3.56,0 -6.524,-1.224 -8.89,-3.671c-2.367,-2.447 -3.55,-5.846 -3.55,-10.194c0,-4.349 1.102,-7.727 3.307,-10.134c2.205,-2.407 5.087,-3.611 8.647,-3.611Zm0.061,2.367c-1.092,0 -2.033,0.273 -2.822,0.819c-0.789,0.546 -1.183,1.325 -1.183,2.336c-0,2.589 1.011,5.067 3.034,7.434c2.023,2.366 4.43,3.934 7.221,4.703c0.081,-1.538 0.121,-2.852 0.121,-3.945c0,-4.207 -0.445,-7.15 -1.335,-8.829c-0.89,-1.679 -2.568,-2.518 -5.036,-2.518Zm-6.25,11.347c-0,4.127 0.516,7.08 1.547,8.86c1.032,1.78 2.721,2.67 5.067,2.67c1.618,-0 2.862,-0.415 3.732,-1.244c0.87,-0.83 1.466,-2.175 1.79,-4.036c-2.63,-0.566 -5.047,-1.759 -7.251,-3.58c-2.205,-1.82 -3.813,-3.843 -4.825,-6.068c-0.04,0.688 -0.06,1.821 -0.06,3.398Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/><path
|
||||||
|
d="M442.571,112.75c0.729,-0 1.093,1.072 1.093,3.216c-0,3.438 -0.739,6.189 -2.215,8.252c-1.477,2.064 -3.631,3.095 -6.463,3.095c-2.832,0 -5.006,-0.971 -6.523,-2.912c-1.517,-1.942 -2.275,-4.693 -2.275,-8.253l-0,-7.1c-0,-4.814 -1.214,-7.221 -3.641,-7.221c-2.144,0 -3.904,1.093 -5.28,3.277c-1.375,2.185 -2.063,5.846 -2.063,10.983l0,10.377c-0.404,0.566 -1.112,0.849 -2.124,0.849c-1.011,0 -1.82,-0.313 -2.427,-0.94c-0.607,-0.627 -0.91,-1.548 -0.91,-2.761l0,-23.059c0.405,-0.566 1.173,-0.85 2.306,-0.85c2.103,0 3.155,1.517 3.155,4.552c1.699,-3.035 4.39,-4.552 8.071,-4.552c2.629,0 4.723,0.83 6.28,2.488c1.558,1.659 2.337,4.046 2.337,7.161l-0,8.616c-0,2.306 0.364,4.046 1.092,5.219c0.728,1.173 1.81,1.76 3.246,1.76c1.436,-0 2.7,-0.9 3.793,-2.701c1.092,-1.8 1.638,-4.844 1.638,-9.132c0.041,-0.243 0.344,-0.364 0.91,-0.364Z"
|
||||||
|
style="fill-rule:nonzero;"
|
||||||
|
/></g
|
||||||
|
></g
|
||||||
|
></svg
|
||||||
|
>
|
||||||
|
</a>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
a {
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
header {
|
||||||
|
text-align: center;
|
||||||
|
margin-top: 3rem;
|
||||||
|
margin-bottom: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 30rem) {
|
||||||
|
header {
|
||||||
|
margin-top: 1rem;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
header svg {
|
||||||
|
max-height: 4rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
header svg {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 16rem;
|
||||||
|
transform: translateX(-1rem);
|
||||||
|
fill: currentColor;
|
||||||
|
}
|
||||||
|
</style>
|
42
frontend/src/routes/__layout.svelte
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
<script lang="ts" context="module">
|
||||||
|
import { init, waitLocale, getLocaleFromNavigator } from 'svelte-intl-precompile'
|
||||||
|
// @ts-ignore
|
||||||
|
import { registerAll } from '$locales'
|
||||||
|
registerAll()
|
||||||
|
init({ initialLocale: getLocaleFromNavigator() ?? undefined, fallbackLocale: 'en' })
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { init as initStores } from '$lib/stores/status'
|
||||||
|
import Footer from '$lib/views/Footer.svelte'
|
||||||
|
import Header from '$lib/views/Header.svelte'
|
||||||
|
import { onMount } from 'svelte'
|
||||||
|
import '../app.css'
|
||||||
|
|
||||||
|
onMount(() => {
|
||||||
|
initStores()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<svelte:head>
|
||||||
|
<title>cryptgeon</title>
|
||||||
|
</svelte:head>
|
||||||
|
|
||||||
|
{#await waitLocale() then _}
|
||||||
|
<main>
|
||||||
|
<Header />
|
||||||
|
<slot />
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<Footer />
|
||||||
|
{/await}
|
||||||
|
|
||||||
|
<style>
|
||||||
|
main {
|
||||||
|
padding: 1rem;
|
||||||
|
padding-bottom: 4rem;
|
||||||
|
width: 100%;
|
||||||
|
max-width: 35rem;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
</style>
|
87
frontend/src/routes/about.svelte
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
<script context="module">
|
||||||
|
import { browser, dev } from '$app/env'
|
||||||
|
import { status } from '$lib/stores/status'
|
||||||
|
import AboutParagraph from '$lib/ui/AboutParagraph.svelte'
|
||||||
|
|
||||||
|
export const hydrate = dev
|
||||||
|
export const router = browser
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<svelte:head>
|
||||||
|
<title>About</title>
|
||||||
|
</svelte:head>
|
||||||
|
|
||||||
|
<section class="content">
|
||||||
|
<h1>About</h1>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<i>cryptgeon</i> is a secure, open source sharing note / file service inspired by
|
||||||
|
<a href="https://privnote.com"><i>PrivNote</i></a>.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<AboutParagraph title="how does it work?">
|
||||||
|
<span>
|
||||||
|
each note has a generated <code>id (256bit)</code> and <code>key 256(bit)</code>. The
|
||||||
|
<code>id</code>
|
||||||
|
is used to save & retrieve the note. the note is then encrypted with aes in gcm mode on the client
|
||||||
|
side with the <code>key</code> and then sent to the server. data is stored in memory and never
|
||||||
|
persisted to disk. the server never sees the encryption key and cannot decrypt the contents of
|
||||||
|
the notes even if it tried to.
|
||||||
|
</span>
|
||||||
|
</AboutParagraph>
|
||||||
|
|
||||||
|
<AboutParagraph title="features">
|
||||||
|
<ul>
|
||||||
|
<li>server cannot decrypt contents due to client side encryption</li>
|
||||||
|
<li>view and time constraints</li>
|
||||||
|
<li>in memory, no persistence</li>
|
||||||
|
</ul>
|
||||||
|
</AboutParagraph>
|
||||||
|
|
||||||
|
<AboutParagraph title="tech stack">
|
||||||
|
<span>
|
||||||
|
the backend is written in rust and the frontend is svelte and typescript.
|
||||||
|
<br />
|
||||||
|
you are welcomed to check & audit the
|
||||||
|
<a href="https://github.com/cupcakearmy/cryptgeon" target="_blank" rel="noopener">
|
||||||
|
source code
|
||||||
|
</a>.
|
||||||
|
</span>
|
||||||
|
</AboutParagraph>
|
||||||
|
|
||||||
|
<AboutParagraph title="translations">
|
||||||
|
<span
|
||||||
|
>translations are managed on <a href="https://lokalise.com/" target="_blank">Lokalise</a>,
|
||||||
|
which granted an open source license to use the paid version. If you are interested in helping
|
||||||
|
translating don't hesitate to contact me!
|
||||||
|
</span>
|
||||||
|
</AboutParagraph>
|
||||||
|
|
||||||
|
<AboutParagraph title="attribution">
|
||||||
|
<span>
|
||||||
|
icons made by <a href="https://www.freepik.com" title="Freepik">freepik</a> from
|
||||||
|
<a href="https://www.flaticon.com/" title="Flaticon">www.flaticon.com</a>
|
||||||
|
</span>
|
||||||
|
</AboutParagraph>
|
||||||
|
|
||||||
|
<AboutParagraph title="version">
|
||||||
|
<span>
|
||||||
|
{#if $status}
|
||||||
|
<code>v{$status.version}</code>
|
||||||
|
{/if}
|
||||||
|
</span>
|
||||||
|
</AboutParagraph>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
section {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
padding-left: 1rem;
|
||||||
|
list-style: square;
|
||||||
|
}
|
||||||
|
</style>
|
@@ -1,7 +1,3 @@
|
|||||||
<script context="module" lang="ts">
|
|
||||||
export const prerender = true
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import Create from '$lib/views/Create.svelte'
|
import Create from '$lib/views/Create.svelte'
|
||||||
</script>
|
</script>
|
83
frontend/src/routes/note/[id].svelte
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
<script context="module" lang="ts">
|
||||||
|
import type { Load } from '@sveltejs/kit'
|
||||||
|
|
||||||
|
export const load: Load = async ({ params }) => {
|
||||||
|
return {
|
||||||
|
props: params,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { onMount } from 'svelte'
|
||||||
|
import { t } from 'svelte-intl-precompile'
|
||||||
|
|
||||||
|
import type { NotePublic } from '$lib/api'
|
||||||
|
import { get, info } from '$lib/api'
|
||||||
|
import { decrypt, getKeyFromString } from '$lib/crypto'
|
||||||
|
import Button from '$lib/ui/Button.svelte'
|
||||||
|
import ShowNote from '$lib/ui/ShowNote.svelte'
|
||||||
|
|
||||||
|
export let id: string
|
||||||
|
|
||||||
|
let password: string
|
||||||
|
let note: NotePublic | null = null
|
||||||
|
let exists = false
|
||||||
|
|
||||||
|
let loading = true
|
||||||
|
let error = false
|
||||||
|
|
||||||
|
onMount(async () => {
|
||||||
|
try {
|
||||||
|
loading = true
|
||||||
|
error = false
|
||||||
|
password = window.location.hash.slice(1)
|
||||||
|
await info(id)
|
||||||
|
exists = true
|
||||||
|
} catch {
|
||||||
|
exists = false
|
||||||
|
} finally {
|
||||||
|
loading = false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
async function show() {
|
||||||
|
try {
|
||||||
|
error = false
|
||||||
|
loading = true
|
||||||
|
const data = note || (await get(id)) // Don't get the content twice on wrong password.
|
||||||
|
const key = await getKeyFromString(password)
|
||||||
|
data.contents = await decrypt(data.contents, key)
|
||||||
|
note = data
|
||||||
|
} catch {
|
||||||
|
error = true
|
||||||
|
} finally {
|
||||||
|
loading = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{#if !loading}
|
||||||
|
{#if !exists}
|
||||||
|
<p class="error-text">{$t('show.errors.not_found')}</p>
|
||||||
|
{:else if note && !error}
|
||||||
|
<ShowNote {note} />
|
||||||
|
{:else}
|
||||||
|
<form on:submit|preventDefault={show}>
|
||||||
|
<fieldset>
|
||||||
|
<p>{$t('show.explanation')}</p>
|
||||||
|
<Button type="submit">{$t('show.show_note')}</Button>
|
||||||
|
{#if error}
|
||||||
|
<br />
|
||||||
|
<p class="error-text">
|
||||||
|
{$t('show.errors.decryption_failed')}
|
||||||
|
<br />
|
||||||
|
</p>
|
||||||
|
{/if}
|
||||||
|
</fieldset>
|
||||||
|
</form>
|
||||||
|
{/if}
|
||||||
|
{/if}
|
||||||
|
{#if loading}
|
||||||
|
<p>{$t('common.loading')}</p>
|
||||||
|
{/if}
|
Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 6.1 KiB |
@@ -1,3 +1,4 @@
|
|||||||
# https://www.robotstxt.org/robotstxt.html
|
# https://www.robotstxt.org/robotstxt.html
|
||||||
User-agent: *
|
User-agent: *
|
||||||
Disallow:
|
Allow: /$
|
||||||
|
Disallow: /
|
18
frontend/svelte.config.js
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
import preprocess from 'svelte-preprocess'
|
||||||
|
import adapter from '@sveltejs/adapter-static'
|
||||||
|
import precompileIntl from 'svelte-intl-precompile/sveltekit-plugin'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
preprocess: preprocess(),
|
||||||
|
|
||||||
|
kit: {
|
||||||
|
adapter: adapter({
|
||||||
|
fallback: 'index.html',
|
||||||
|
}),
|
||||||
|
vite: {
|
||||||
|
plugins: [
|
||||||
|
precompileIntl('locales'), // if your translations are defined in /locales/[lang].json
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|