Compare commits
9 Commits
main
...
09916187d5
Author | SHA1 | Date | |
---|---|---|---|
09916187d5
|
|||
64a709e5a0
|
|||
f9e4866597
|
|||
d532d51ab9
|
|||
1512a1452f
|
|||
199a908ead
|
|||
0403ba0bc4
|
|||
e7471b20e4
|
|||
4fa3212bdd
|
119
README.md
119
README.md
@@ -0,0 +1,119 @@
|
||||
# atproto
|
||||
|
||||
https://github.com/bluesky-social/atproto
|
||||
|
||||
- @
|
||||
- [at]mosphere
|
||||
- at://proto
|
||||
- at protocol
|
||||
|
||||
## account
|
||||
|
||||
[@ai.syu.is](https://web.syu.is/profile/ai.syu.is)
|
||||
|
||||
- https://at.syu.is/at/yui.syui.ai
|
||||
- https://plc.syu.is/did:plc:6qyecktefllvenje24fcxnie
|
||||
- https://plc.directory/did:plc:ytvoptig4ddshmwdsjmhtcym
|
||||
|
||||
```sh
|
||||
$ curl -sL syu.is/xrpc/_health
|
||||
{"version":"0.4.65"}
|
||||
|
||||
# latest
|
||||
# https://github.com/bluesky-social/atproto/blob/main/packages/pds/package.json
|
||||
$ curl -sL https://raw.githubusercontent.com/bluesky-social/atproto/refs/heads/main/packages/pds/package.json |jq -r .version
|
||||
```
|
||||
|
||||
```sh
|
||||
$ curl -sL "syu.is/xrpc/com.atproto.repo.describeRepo?repo=ai.syu.is" |jq -r .did
|
||||
did:plc:6qyecktefllvenje24fcxnie
|
||||
|
||||
$ curl -sL "syu.is/xrpc/com.atproto.repo.listRecords?repo=ai.syu.is&collection=app.bsky.feed.post&reverse=true&limit=1"
|
||||
{"records":[{"uri":"at://did:plc:6qyecktefllvenje24fcxnie/app.bsky.feed.post/3l6s2riuouk2j","cid":"bafyreibjohl7va4upkibw5twaxdd4jg3l6rmfatu4dpjjfd5xkb2ijtlx4","value":{"text":"hello","$type":"app.bsky.feed.post","langs":["ja"],"createdAt":"2024-10-18T13:21:39.809Z"}}],"cursor":"3l6s2riuouk2j"}
|
||||
```
|
||||
|
||||
## feed
|
||||
|
||||
> at://did:plc:6qyecktefllvenje24fcxnie/app.bsky.feed.generator/cmd
|
||||
|
||||
- https://api.bsky.app/xrpc/app.bsky.feed.getFeed?feed=at://did:plc:4hqjfn7m6n5hno3doamuhgef/app.bsky.feed.generator/cmd
|
||||
- https://feed.syu.is/xrpc/app.bsky.feed.getFeedSkeleton?feed=at://did:plc:4hqjfn7m6n5hno3doamuhgef/app.bsky.feed.generator/cmd
|
||||
- https://desc.syu.is/xrpc/app.bsky.feed.getFeedSkeleton?feed=at://did:plc:6qyecktefllvenje24fcxnie/app.bsky.feed.generator/cmd
|
||||
|
||||
## link
|
||||
|
||||
- https://github.com/bluesky-social/atproto
|
||||
- https://github.com/itaru2622/bluesky-selfhost-env
|
||||
- https://github.com/bluesky-social/atproto/discussions/2026
|
||||
|
||||
## self-host
|
||||
|
||||
currently, bsky and bsync require patches to function properly. additionally, social-app is not displaying avatars. for components that are not working, it's recommended to use [itaru2622/bluesky-selfhost-env](https://github.com/itaru2622/bluesky-selfhost-env). this repository provides an environment for self-hosting bluesky.
|
||||
|
||||
- bsky = appview
|
||||
- ozone = mod
|
||||
|
||||
|name|service|patch|
|
||||
|---|---|---|
|
||||
|pds|https://github.com/bluesky-social/atproto/blob/main/services/pds/Dockerfile||
|
||||
|bsky|https://github.com/bluesky-social/atproto/blob/main/services/bsky/Dockerfile|[itaru2622/bluesky-atproto-bsky](https://hub.docker.com/r/itaru2622/bluesky-atproto-bsky)|
|
||||
|bsync|https://github.com/bluesky-social/atproto/blob/main/services/bsync/Dockerfile||
|
||||
|ozone|https://github.com/bluesky-social/atproto/blob/main/services/ozone/Dockerfile||
|
||||
|plc|https://github.com/did-method-plc/did-method-plc/tree/main/packages/server||
|
||||
|bgs|https://github.com/bluesky-social/indigo/tree/main/cmd/bigsky||
|
||||
|feed|https://github.com/bluesky-social/feed-generator||
|
||||
|web|https://github.com/bluesky-social/social-app|[bluesky-selfhost-env](https://github.com/itaru2622/bluesky-selfhost-env/blob/master/patching/160-social-app-disable-hackModifyThumbnailPath.diff)|
|
||||
|
||||
```sh
|
||||
# BSKY_IMG_URI_ENDPOINT, BSKY_BLOB_CACHE_LOC
|
||||
# avatar link example
|
||||
1. https://appview.${host}/img/avatar/plain/${did}/${cid}@jpeg
|
||||
2. https://${host}/xrpc/com.atproto.sync.getBlob?did=${did}&cid=${cid}
|
||||
```
|
||||
|
||||
docker compose will not be published unless you write ports. it is only valid internally. add ports only for what you want to publish.
|
||||
|
||||
## api
|
||||
|
||||
```sh
|
||||
# create account
|
||||
url=https://${pds}/xrpc/com.atproto.server.createAccount
|
||||
json="{\"email\": \"$email\", \"handle\": \"$handle\", \"password\": \"$password\"}"
|
||||
curl -X POST -H "Content-Type: application/json" -d $json -sL $url
|
||||
```
|
||||
|
||||
change `src/pds/handle` to use a name of 3 characters or less. also, you cannot create an account with a name of 3 characters or less from social-app (web client). please create it from api.
|
||||
|
||||
- [/atproto/packages/pds/src/handle/index.ts](https://github.com/bluesky-social/atproto/blame/d4d5a6edba972c0e9976289bde8bc0b42ff547ca/packages/pds/src/handle/index.ts#L86-L88)
|
||||
|
||||
```sh
|
||||
# invite code
|
||||
admin_password=xxx
|
||||
url=https://$host/xrpc/com.atproto.server.createInviteCode
|
||||
json="{\"useCount\":1}"
|
||||
curl -X POST -u admin:${admin_password} -H "Content-Type: application/json" -d "$json" -sL $url
|
||||
```
|
||||
|
||||
## oauth
|
||||
|
||||
```sh
|
||||
# https://github.com/bluesky-social/cookbook/tree/main/python-oauth-web-app
|
||||
$ cd ./repos/cookbook/python-oauth-web-app
|
||||
$ rye sync
|
||||
$ rye run python3 -c 'import secrets; print(secrets.token_hex())'|xargs echo FLASK_SECRET_KEY|tr -d ' ' >> .env
|
||||
$ rye run python3 generate_jwk.py |xargs echo FLASK_CLIENT_SECRET_JWK|tr -d ' ' >> .env
|
||||
$ cat .env
|
||||
$ rye run flask run
|
||||
```
|
||||
|
||||
please access `127.0.0.1:5000`. it may not work if you use localhost.
|
||||
|
||||
also, oauth doesn't work on localhost. use [ngrok](https://ngrok.com/), [tailscale](https://tailscale.com/), [cloudflare](https://github.com/cloudflare/cloudflared).
|
||||
|
||||
```sh
|
||||
$ ngrok http http://localhost:5000
|
||||
```
|
||||
|
||||
```sh
|
||||
$ cloudflared tunnel --url http://localhost:5000
|
||||
```
|
||||
|
47
at.json
Normal file
47
at.json
Normal file
@@ -0,0 +1,47 @@
|
||||
{
|
||||
"atmosphere": {
|
||||
"name": "at",
|
||||
"repo": "https://github.com/bluesky-social/atproto",
|
||||
"uri": "https://atproto.com/ja/guides/glossary",
|
||||
"exosphere": {
|
||||
"km":[{ "min": 700, "max": 10000 }],
|
||||
"tag": [ "universe" ]
|
||||
},
|
||||
"thermosphere": {
|
||||
"km":[{ "min": 80, "max": 700 }],
|
||||
"tag": [ "aurora" ]
|
||||
},
|
||||
"mesosphere": {
|
||||
"km":[{ "min": 50, "max": 80 }],
|
||||
"tag": [ "meteor" ]
|
||||
},
|
||||
"stratosphere": {
|
||||
"km":[{ "min": 12, "max": 50 }],
|
||||
"tag": [ "ozone", "bigsky" ],
|
||||
"service":[
|
||||
{ "name":"ozone", "repo":"https://github.com/bluesky-social/atproto/tree/main/services/ozone" },
|
||||
{ "name":"bgs", "repo":"https://github.com/bluesky-social/indigo/tree/main/cmd/bigsky" }
|
||||
]
|
||||
},
|
||||
"troposphere": {
|
||||
"km":[{ "min": 0, "max": 12 }],
|
||||
"tag": [ "bluesky"],
|
||||
"service":[
|
||||
{ "name":"bsky", "repo":"https://github.com/bluesky-social/atproto/tree/main/services/bsky", "tag":[ "api", "appview" ] },
|
||||
{ "name":"bsync","repo":"https://github.com/bluesky-social/atproto/tree/main/services/bsync" },
|
||||
{ "name":"pds", "repo":"https://github.com/bluesky-social/atproto/tree/main/services/pds" }
|
||||
]
|
||||
},
|
||||
"other": {
|
||||
"tag": [ "plc", "feed", "oauth", "social-app", "stream" ],
|
||||
"service":[
|
||||
{ "name":"plc", "repo":"https://github.com/did-method-plc/did-method-plc/tree/main/packages/server", "tag" : [ "did" ] },
|
||||
{ "name":"social-app", "repo":"https://github.com/bluesky-social/social-app", "tag": [ "web" ] },
|
||||
{ "name":"oauth", "repo":"https://github.com/bluesky-social/cookbook/tree/main/python-oauth-web-app" },
|
||||
{ "name":"feed", "repo":"https://github.com/bluesky-social/feed-generator" },
|
||||
{ "name":"stream", "repo":"https://github.com/bluesky-social/jetstream" }
|
||||
]
|
||||
}
|
||||
},
|
||||
"ref": "https://en.wikipedia.org/wiki/Atmosphere_of_Earth"
|
||||
}
|
30
build.sh
Executable file
30
build.sh
Executable file
@@ -0,0 +1,30 @@
|
||||
#!/bin/zsh
|
||||
|
||||
d=${0:a:h}
|
||||
cd $d/repos
|
||||
|
||||
t="
|
||||
https://github.com/bluesky-social/atproto
|
||||
https://github.com/did-method-plc/did-method-plc
|
||||
https://github.com/bluesky-social/feed-generator
|
||||
https://github.com/bluesky-social/social-app
|
||||
https://github.com/bluesky-social/cookbook
|
||||
https://github.com/itaru2622/bluesky-selfhost-env
|
||||
"
|
||||
|
||||
t=`echo $t|grep -v '^$'`
|
||||
n=`echo $t|wc -l`
|
||||
|
||||
for ((i=1;i<=$n;i++))
|
||||
do
|
||||
tt=`echo $t|awk "NR==$i"`
|
||||
dd=$d/repos/$tt:t
|
||||
if [ -d $dd ];then
|
||||
echo ok
|
||||
cd $dd
|
||||
git pull
|
||||
cd $d/repos
|
||||
else
|
||||
git clone $tt
|
||||
fi
|
||||
done
|
166
compose.yaml
Normal file
166
compose.yaml
Normal file
@@ -0,0 +1,166 @@
|
||||
services:
|
||||
|
||||
database:
|
||||
image: postgres:16-alpine
|
||||
restart: always
|
||||
env_file:
|
||||
- ./envs/postgres
|
||||
volumes:
|
||||
- ./configs/postgres/init/:/docker-entrypoint-initdb.d/
|
||||
- ./data/postgres/:/var/lib/postgresql/data/
|
||||
healthcheck:
|
||||
test: "pg_isready -U $$POSTGRES_USER -d $$POSTGRES_DB"
|
||||
interval: 5s
|
||||
retries: 20
|
||||
|
||||
redis:
|
||||
image: redis:alpine
|
||||
restart: always
|
||||
volumes:
|
||||
- ./data/redis/:/data/
|
||||
healthcheck:
|
||||
test: ["CMD", "redis-cli", "ping", "|", "grep", "PONG"]
|
||||
interval: 1s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
|
||||
pds-fix-permission:
|
||||
image: alpine:latest
|
||||
volumes:
|
||||
- ./data/pds/:/data/
|
||||
command: chown 1000.1000 /data
|
||||
|
||||
bsky-fix-permission:
|
||||
image: alpine:latest
|
||||
volumes:
|
||||
- ./data/bsky/:/data/
|
||||
command: chown 1000.1000 /data
|
||||
|
||||
plc:
|
||||
ports:
|
||||
- 2582:3000
|
||||
build:
|
||||
context: ./repos/did-method-plc/
|
||||
dockerfile: packages/server/Dockerfile
|
||||
restart: always
|
||||
env_file:
|
||||
- ./envs/plc
|
||||
depends_on:
|
||||
database:
|
||||
condition: service_healthy
|
||||
|
||||
pds:
|
||||
ports:
|
||||
- 2583:3000
|
||||
#image: itaru2622/bluesky-atproto-pds
|
||||
#image: ghcr.io/bluesky-social/pds:latest
|
||||
build:
|
||||
context: ./repos/atproto/
|
||||
dockerfile: services/pds/Dockerfile
|
||||
restart: always
|
||||
env_file:
|
||||
- ./envs/pds
|
||||
volumes:
|
||||
- ./data/pds/:/data/
|
||||
depends_on:
|
||||
database:
|
||||
condition: service_healthy
|
||||
pds-fix-permission:
|
||||
condition: service_completed_successfully
|
||||
|
||||
bgs:
|
||||
ports:
|
||||
- 2470:2470
|
||||
build:
|
||||
context: ./repos/indigo/
|
||||
dockerfile: cmd/bigsky/Dockerfile
|
||||
restart: always
|
||||
env_file:
|
||||
- ./envs/bgs
|
||||
volumes:
|
||||
- ./data/bgs/:/data/
|
||||
depends_on:
|
||||
database:
|
||||
condition: service_healthy
|
||||
|
||||
ozone:
|
||||
ports:
|
||||
- 2585:3000
|
||||
build:
|
||||
context: ./repos/atproto/
|
||||
dockerfile: services/ozone/Dockerfile
|
||||
restart: always
|
||||
command: node --enable-source-maps api.js
|
||||
volumes:
|
||||
- ./data/ozone/:/data/
|
||||
- ./repos/ozone.js:/app/services/ozone/api.js:ro
|
||||
env_file:
|
||||
- ./envs/ozone
|
||||
healthcheck:
|
||||
test: "wget -q --spider http://localhost:3000/xrpc/_health"
|
||||
interval: 5s
|
||||
retries: 20
|
||||
depends_on:
|
||||
database:
|
||||
condition: service_healthy
|
||||
|
||||
ozone-daemon:
|
||||
build:
|
||||
context: ./repos/atproto/
|
||||
dockerfile: services/ozone/Dockerfile
|
||||
restart: always
|
||||
command: node --enable-source-maps daemon.js
|
||||
env_file:
|
||||
- ./envs/ozone
|
||||
depends_on:
|
||||
ozone:
|
||||
condition: service_healthy
|
||||
database:
|
||||
condition: service_healthy
|
||||
|
||||
social-app:
|
||||
ports:
|
||||
- 8100:8100
|
||||
#image: itaru2622/bluesky-social-app
|
||||
build:
|
||||
context: ./repos/social-app/
|
||||
dockerfile: Dockerfile
|
||||
restart: always
|
||||
env_file:
|
||||
- ./envs/social-app
|
||||
command: "/usr/bin/bskyweb serve"
|
||||
|
||||
bsky:
|
||||
ports:
|
||||
- 2584:2584
|
||||
image: itaru2622/bluesky-atproto-bsky
|
||||
#build:
|
||||
# context: ./repos/atproto/
|
||||
# dockerfile: services/bsky/Dockerfile
|
||||
restart: always
|
||||
env_file:
|
||||
- ./envs/bsky
|
||||
user: root
|
||||
volumes:
|
||||
- ./data/bsky/:/data/
|
||||
# - ./repos/bsky.js:/app/services/bsky/api.js:ro
|
||||
command: node --enable-source-maps api.js
|
||||
depends_on:
|
||||
database:
|
||||
condition: service_healthy
|
||||
redis:
|
||||
condition: service_healthy
|
||||
bsky-fix-permission:
|
||||
condition: service_completed_successfully
|
||||
|
||||
feed:
|
||||
ports:
|
||||
- 2586:2586
|
||||
build:
|
||||
context: ./repos/feed-generator/
|
||||
restart: always
|
||||
env_file:
|
||||
- ./envs/feed
|
||||
volumes:
|
||||
- ./data/feed/:/data/
|
||||
|
35
configs/postgres/init/init.sql
Normal file
35
configs/postgres/init/init.sql
Normal file
@@ -0,0 +1,35 @@
|
||||
-- PLC
|
||||
CREATE DATABASE plc;
|
||||
GRANT ALL PRIVILEGES ON DATABASE plc TO postgres;
|
||||
|
||||
-- BGS
|
||||
CREATE DATABASE bgs;
|
||||
CREATE DATABASE carstore;
|
||||
GRANT ALL PRIVILEGES ON DATABASE bgs TO postgres;
|
||||
GRANT ALL PRIVILEGES ON DATABASE carstore TO postgres;
|
||||
|
||||
-- bsky(AppView)
|
||||
--CREATE DATABASE appview;
|
||||
--GRANT ALL PRIVILEGES ON DATABASE appview TO postgres;
|
||||
CREATE DATABASE bsky;
|
||||
GRANT ALL PRIVILEGES ON DATABASE bsky TO postgres;
|
||||
|
||||
-- ozone(Moderation)
|
||||
--CREATE DATABASE moderation;
|
||||
--GRANT ALL PRIVILEGES ON DATABASE moderation TO postgres;
|
||||
CREATE DATABASE ozone;
|
||||
GRANT ALL PRIVILEGES ON DATABASE ozone TO postgres;
|
||||
|
||||
-- search(palomar)
|
||||
CREATE DATABASE search;
|
||||
GRANT ALL PRIVILEGES ON DATABASE search TO postgres;
|
||||
--CREATE DATABASE palomar;
|
||||
--GRANT ALL PRIVILEGES ON DATABASE palomar TO postgres;
|
||||
|
||||
-- PDS
|
||||
CREATE DATABASE pds;
|
||||
GRANT ALL PRIVILEGES ON DATABASE pds TO postgres;
|
||||
|
||||
-- BSYNC
|
||||
CREATE DATABASE bsync;
|
||||
GRANT ALL PRIVILEGES ON DATABASE bsync TO postgres;
|
6
envs/bgs
Normal file
6
envs/bgs
Normal file
@@ -0,0 +1,6 @@
|
||||
DATABASE_URL=postgres://postgres:postgres@database/bgs
|
||||
CARSTORE_DATABASE_URL=postgres://postgres:postgres@database/carstore
|
||||
DATA_DIR=/data
|
||||
ATP_PLC_HOST=https://plc.${host}
|
||||
|
||||
BGS_ADMIN_KEY
|
19
envs/bsky
Normal file
19
envs/bsky
Normal file
@@ -0,0 +1,19 @@
|
||||
BSKY_PORT=2584
|
||||
BSKY_BLOB_CACHE_LOC=/data/
|
||||
BSKY_BSYNC_HTTP_VERSION=1.1
|
||||
BSKY_BSYNC_PORT=3002
|
||||
BSKY_BSYNC_URL=http://bsky:3002
|
||||
BSKY_COURIER_URL=http://fake-courier.example.invalid/
|
||||
BSKY_DATAPLANE_HTTP_VERSION=1.1
|
||||
BSKY_DATAPLANE_PORT=3001
|
||||
BSKY_DATAPLANE_URLS=http://bsky:3001
|
||||
BSKY_DB_POSTGRES_URL=postgres://postgres:postgres@database/bsky
|
||||
BSKY_DID_PLC_URL=https://plc.${host}
|
||||
BSKY_PUBLIC_URL=https://bsky.${host}
|
||||
BSKY_REPO_PROVIDER=wss://bgs.${host}
|
||||
BSKY_SERVER_DID=did:web:bsky.${host}
|
||||
MOD_SERVICE_DID=did:web:ozone.${host}
|
||||
|
||||
#BSKY_IMG_URI_ENDPOINT=https://bsky.${host}/img
|
||||
BSKY_ADMIN_PASSWORDS
|
||||
BSKY_SERVICE_SIGNING_KEY
|
21
envs/ozone
Normal file
21
envs/ozone
Normal file
@@ -0,0 +1,21 @@
|
||||
OZONE_SERVER_DID=did:web:ozone.${host}
|
||||
OZONE_PUBLIC_URL=https://ozone.${host}
|
||||
OZONE_ADMIN_HANDLE=${user}.${host}
|
||||
OZONE_MODERATOR_DIDS=${did}
|
||||
OZONE_ADMIN_DIDS=${did}
|
||||
OZONE_DB_POSTGRES_URL=postgres://postgres:postgres@database/ozone
|
||||
OZONE_DID_PLC_URL=https://plc.${host}
|
||||
NEXT_PUBLIC_PLC_DIRECTORY_URL=https://plc.${host}
|
||||
NEXT_PUBLIC_OZONE_SERVICE_DID=did:web:ozone.${host}
|
||||
NEXT_PUBLIC_SOCIAL_APP_DOMAIN=mod.${host}
|
||||
NEXT_PUBLIC_SOCIAL_APP_URL=https://mod.${host}
|
||||
OZONE_APPVIEW_DID=did:web:bsky.${host}
|
||||
OZONE_APPVIEW_URL=https://bsky.${host}
|
||||
OZONE_APPVIEW_PUSH_EVENTS=false
|
||||
OZONE_PDS_DID=did:web:${host}
|
||||
OZONE_PDS_URL=https://${host}
|
||||
OZONE_DEV_MODE=true
|
||||
OZONE_DB_MIGRATE=1
|
||||
|
||||
OZONE_ADMIN_PASSWORD
|
||||
OZONE_SIGNING_KEY_HEX
|
21
envs/pds
Normal file
21
envs/pds
Normal file
@@ -0,0 +1,21 @@
|
||||
PDS_HOSTNAME=${host}
|
||||
PDS_DB_POSTGRES_URL=postgres://postgres:postgres@database/pds
|
||||
PDS_DATA_DIRECTORY=/data
|
||||
PDS_BLOBSTORE_DISK_LOCATION=/data/img/static
|
||||
#PDS_BLOBSTORE_DISK_TMP_LOCATION=/data/img/tmp
|
||||
PDS_BSKY_APP_VIEW_DID=did:web:bsky.${host}
|
||||
PDS_BSKY_APP_VIEW_URL=https://bsky.${host}
|
||||
PDS_CRAWLERS=https://bgs.${host}
|
||||
PDS_DEV_MODE=true
|
||||
PDS_DID_PLC_URL=https://plc.${host}
|
||||
PDS_ENABLE_DID_DOC_WITH_SESSION=true
|
||||
PDS_INVITE_INTERVAL=604800000
|
||||
PDS_SERVICE_DID=did:web:${host}
|
||||
PDS_EMAIL_FROM_ADDRESS=no-reply@${host}
|
||||
|
||||
PDS_INVITE_REQUIRED=true
|
||||
PDS_EMAIL_SMTP_URL=smtps://${user}:${app_password}@smtp.gmail.com
|
||||
PDS_ADMIN_PASSWORD
|
||||
PDS_PLC_ROTATION_KEY_K256_PRIVATE_KEY_HEX
|
||||
PDS_REPO_SIGNING_KEY_K256_PRIVATE_KEY_HEX
|
||||
PDS_JWT_SECRET
|
4
envs/plc
Normal file
4
envs/plc
Normal file
@@ -0,0 +1,4 @@
|
||||
DATABASE_URL=postgres://postgres:postgres@database/plc
|
||||
DB_CREDS_JSON='{"username":"postgres","password":"postgres","host":"database","port":"5432","database":"plc"}'
|
||||
ENABLE_MIGRATIONS=true
|
||||
DB_MIGRATE_CREDS_JSON='{"username":"postgres","password":"postgres","host":"database","port":"5432","database":"plc"}'
|
4
envs/postgres
Normal file
4
envs/postgres
Normal file
@@ -0,0 +1,4 @@
|
||||
#POSTGRES_INITDB_ARGS=--encoding=UTF-8 --lc-collate=C --lc-ctype=C
|
||||
POSTGRES_USER=postgres
|
||||
POSTGRES_PASSWORD=postgres
|
||||
POSTGRES_DB=healthcheck
|
1
envs/social-app
Normal file
1
envs/social-app
Normal file
@@ -0,0 +1 @@
|
||||
ATP_APPVIEW_HOST=https://bsky.${host}
|
52
icons/Logotype.tsx
Normal file
52
icons/Logotype.tsx
Normal file
@@ -0,0 +1,52 @@
|
||||
import React from 'react'
|
||||
import Svg, {Path, SvgProps, PathProps} from 'react-native-svg'
|
||||
|
||||
import {usePalette} from '#/lib/hooks/usePalette'
|
||||
|
||||
const ratio = 17 / 64
|
||||
|
||||
export function Logotype({
|
||||
fill,
|
||||
...rest
|
||||
}: {fill?: PathProps['fill']} & SvgProps) {
|
||||
const pal = usePalette('default')
|
||||
// @ts-ignore it's fiiiiine
|
||||
const size = parseInt(rest.width || 32)
|
||||
|
||||
return (
|
||||
<Svg
|
||||
fill="none"
|
||||
viewBox="0 0 2821.6379 794.29016"
|
||||
{...rest}
|
||||
width={size}
|
||||
height={Number(size) * ratio}>
|
||||
<g
|
||||
transform="matrix(0.1,0,0,-0.1,-282.80153,1445)"
|
||||
fill="#000000"
|
||||
stroke="none"
|
||||
>
|
||||
<path
|
||||
d="m 24787,14443 c -4,-3 -7,-224 -7,-490 v -483 h 545 545 v 490 490 h -538 c -296,0 -542,-3 -545,-7 z"
|
||||
/>
|
||||
<path
|
||||
d="m 5190,13285 c -8,-3 -96,-12 -195,-20 -199,-16 -296,-32 -430,-70 -49,-14 -115,-32 -145,-40 -153,-39 -504,-198 -662,-301 -21,-13 -57,-36 -80,-51 -24,-16 -72,-50 -109,-78 -241,-182 -377,-315 -528,-517 -119,-158 -120,-160 -106,-188 23,-45 140,-140 560,-457 110,-84 319,-242 465,-353 146,-110 369,-279 495,-375 791,-600 723,-549 1049,-799 269,-207 398,-307 524,-403 l 114,-86 -49,-49 c -128,-131 -378,-258 -588,-299 -66,-13 -357,-13 -420,0 -115,23 -172,39 -202,54 -18,10 -37,17 -43,17 -24,0 -171,81 -255,141 -50,35 -146,121 -215,192 -69,70 -133,127 -142,127 -19,0 -153,-63 -177,-83 -9,-7 -54,-35 -101,-62 -47,-26 -110,-64 -140,-85 -30,-20 -97,-61 -148,-91 -51,-30 -107,-62 -124,-72 -18,-10 -57,-38 -87,-61 -70,-53 -252,-168 -446,-281 -82,-49 -158,-95 -168,-104 -16,-16 -14,-21 34,-106 165,-289 544,-666 867,-860 9,-6 35,-22 58,-37 36,-23 267,-138 349,-173 108,-47 160,-67 240,-93 150,-48 201,-62 228,-62 14,0 64,-9 109,-19 197,-46 302,-56 573,-56 197,1 281,5 345,17 47,9 110,19 141,22 31,3 75,13 99,21 23,8 56,15 72,15 17,0 51,7 77,15 25,8 80,24 121,36 41,12 125,41 185,66 61,25 124,50 140,56 17,7 89,42 160,78 113,58 177,98 395,246 82,56 273,232 403,371 127,136 237,271 237,291 0,12 -208,179 -425,342 -186,140 -1121,843 -1720,1294 -264,199 -589,444 -723,546 -134,101 -274,208 -312,237 -39,29 -70,58 -70,66 0,21 107,115 203,179 95,64 237,133 295,143 20,4 49,12 63,20 96,48 519,48 619,-1 14,-7 41,-16 60,-20 65,-13 262,-118 360,-191 99,-74 250,-230 372,-384 34,-44 70,-80 80,-80 9,0 39,15 67,33 28,17 70,44 94,58 23,15 121,76 217,136 96,60 254,156 350,213 96,56 272,160 390,230 118,70 230,135 248,144 41,21 41,45 -3,116 -18,30 -46,75 -61,100 -64,106 -252,348 -352,454 -106,112 -169,171 -293,274 -197,164 -310,235 -594,376 -215,106 -519,205 -735,240 -137,22 -574,51 -610,41 z"
|
||||
/>
|
||||
<path
|
||||
d="m 29095,12736 c -421,-44 -744,-157 -975,-342 -259,-207 -396,-446 -465,-809 -16,-84 -23,-301 -14,-425 36,-518 257,-859 694,-1070 166,-80 284,-116 716,-215 449,-103 514,-121 646,-186 218,-107 308,-253 306,-494 -2,-303 -188,-477 -588,-551 -140,-26 -640,-27 -825,-1 -275,38 -486,94 -776,203 -94,35 -174,64 -178,64 -3,0 -6,-175 -6,-389 v -388 l 88,-41 c 404,-187 905,-282 1486,-282 675,0 1154,150 1465,459 196,194 311,434 362,756 20,121 17,476 -5,600 -89,517 -358,800 -945,994 -130,43 -241,71 -616,156 -137,31 -299,73 -360,92 -331,106 -455,246 -455,511 1,249 127,412 387,501 272,92 801,76 1269,-39 116,-28 326,-96 432,-138 l 52,-22 -2,406 -3,406 -66,28 c -154,66 -413,140 -604,174 -279,49 -756,69 -1020,42 z"
|
||||
/>
|
||||
<path
|
||||
d="m 9630,12676 c 0,-2 403,-996 896,-2208 493,-1211 898,-2211 901,-2220 6,-21 -135,-386 -193,-499 -104,-202 -256,-324 -471,-380 -118,-30 -340,-43 -516,-29 -84,6 -185,17 -225,24 -40,7 -75,10 -77,7 -3,-2 -5,-163 -5,-357 v -353 l 63,-30 c 194,-93 493,-138 801,-120 414,23 683,115 937,319 174,140 357,402 474,680 26,62 1945,5159 1945,5167 0,2 -232,2 -516,1 l -517,-3 -556,-1550 c -485,-1350 -560,-1550 -578,-1553 -18,-3 -26,11 -62,105 -23,59 -295,758 -605,1553 l -562,1445 -567,3 c -312,1 -567,0 -567,-2 z"
|
||||
/>
|
||||
<path
|
||||
d="m 15690,10867 c 0,-1970 -1,-1936 56,-2153 128,-497 461,-787 1014,-885 147,-26 440,-36 605,-20 413,40 834,200 1181,447 35,25 73,44 88,44 14,0 26,-1 26,-3 0,-2 20,-94 45,-205 25,-111 45,-204 45,-207 0,-3 209,-5 465,-5 h 465 v 2400 2400 h -535 -535 l -2,-1866 -3,-1866 -115,-50 c -425,-185 -743,-252 -1088,-227 -302,21 -472,109 -562,293 -79,161 -74,11 -77,1969 l -3,1747 h -535 -535 z"
|
||||
/>
|
||||
<path
|
||||
d="m 24785,12668 c -3,-7 -4,-1086 -3,-2398 l 3,-2385 538,-3 537,-2 v 2400 2400 h -535 c -419,0 -537,-3 -540,-12 z"
|
||||
/>
|
||||
<path
|
||||
d="m 21660,8275 v -545 h 565 565 v 545 545 h -565 -565 z"
|
||||
/>
|
||||
</g>
|
||||
</Svg>
|
||||
)
|
||||
}
|
32
icons/title.svg
Normal file
32
icons/title.svg
Normal file
@@ -0,0 +1,32 @@
|
||||
<svg
|
||||
viewBox="0 0 2821.6379 794.29016"
|
||||
preserveAspectRatio="xMidYMid"
|
||||
xmlns:svg="http://www.w3.org/2000/svg">
|
||||
<g
|
||||
transform="matrix(0.1,0,0,-0.1,-282.80153,1445)"
|
||||
fill="#000000"
|
||||
stroke="none"
|
||||
>
|
||||
<path
|
||||
d="m 24787,14443 c -4,-3 -7,-224 -7,-490 v -483 h 545 545 v 490 490 h -538 c -296,0 -542,-3 -545,-7 z"
|
||||
/>
|
||||
<path
|
||||
d="m 5190,13285 c -8,-3 -96,-12 -195,-20 -199,-16 -296,-32 -430,-70 -49,-14 -115,-32 -145,-40 -153,-39 -504,-198 -662,-301 -21,-13 -57,-36 -80,-51 -24,-16 -72,-50 -109,-78 -241,-182 -377,-315 -528,-517 -119,-158 -120,-160 -106,-188 23,-45 140,-140 560,-457 110,-84 319,-242 465,-353 146,-110 369,-279 495,-375 791,-600 723,-549 1049,-799 269,-207 398,-307 524,-403 l 114,-86 -49,-49 c -128,-131 -378,-258 -588,-299 -66,-13 -357,-13 -420,0 -115,23 -172,39 -202,54 -18,10 -37,17 -43,17 -24,0 -171,81 -255,141 -50,35 -146,121 -215,192 -69,70 -133,127 -142,127 -19,0 -153,-63 -177,-83 -9,-7 -54,-35 -101,-62 -47,-26 -110,-64 -140,-85 -30,-20 -97,-61 -148,-91 -51,-30 -107,-62 -124,-72 -18,-10 -57,-38 -87,-61 -70,-53 -252,-168 -446,-281 -82,-49 -158,-95 -168,-104 -16,-16 -14,-21 34,-106 165,-289 544,-666 867,-860 9,-6 35,-22 58,-37 36,-23 267,-138 349,-173 108,-47 160,-67 240,-93 150,-48 201,-62 228,-62 14,0 64,-9 109,-19 197,-46 302,-56 573,-56 197,1 281,5 345,17 47,9 110,19 141,22 31,3 75,13 99,21 23,8 56,15 72,15 17,0 51,7 77,15 25,8 80,24 121,36 41,12 125,41 185,66 61,25 124,50 140,56 17,7 89,42 160,78 113,58 177,98 395,246 82,56 273,232 403,371 127,136 237,271 237,291 0,12 -208,179 -425,342 -186,140 -1121,843 -1720,1294 -264,199 -589,444 -723,546 -134,101 -274,208 -312,237 -39,29 -70,58 -70,66 0,21 107,115 203,179 95,64 237,133 295,143 20,4 49,12 63,20 96,48 519,48 619,-1 14,-7 41,-16 60,-20 65,-13 262,-118 360,-191 99,-74 250,-230 372,-384 34,-44 70,-80 80,-80 9,0 39,15 67,33 28,17 70,44 94,58 23,15 121,76 217,136 96,60 254,156 350,213 96,56 272,160 390,230 118,70 230,135 248,144 41,21 41,45 -3,116 -18,30 -46,75 -61,100 -64,106 -252,348 -352,454 -106,112 -169,171 -293,274 -197,164 -310,235 -594,376 -215,106 -519,205 -735,240 -137,22 -574,51 -610,41 z"
|
||||
/>
|
||||
<path
|
||||
d="m 29095,12736 c -421,-44 -744,-157 -975,-342 -259,-207 -396,-446 -465,-809 -16,-84 -23,-301 -14,-425 36,-518 257,-859 694,-1070 166,-80 284,-116 716,-215 449,-103 514,-121 646,-186 218,-107 308,-253 306,-494 -2,-303 -188,-477 -588,-551 -140,-26 -640,-27 -825,-1 -275,38 -486,94 -776,203 -94,35 -174,64 -178,64 -3,0 -6,-175 -6,-389 v -388 l 88,-41 c 404,-187 905,-282 1486,-282 675,0 1154,150 1465,459 196,194 311,434 362,756 20,121 17,476 -5,600 -89,517 -358,800 -945,994 -130,43 -241,71 -616,156 -137,31 -299,73 -360,92 -331,106 -455,246 -455,511 1,249 127,412 387,501 272,92 801,76 1269,-39 116,-28 326,-96 432,-138 l 52,-22 -2,406 -3,406 -66,28 c -154,66 -413,140 -604,174 -279,49 -756,69 -1020,42 z"
|
||||
/>
|
||||
<path
|
||||
d="m 9630,12676 c 0,-2 403,-996 896,-2208 493,-1211 898,-2211 901,-2220 6,-21 -135,-386 -193,-499 -104,-202 -256,-324 -471,-380 -118,-30 -340,-43 -516,-29 -84,6 -185,17 -225,24 -40,7 -75,10 -77,7 -3,-2 -5,-163 -5,-357 v -353 l 63,-30 c 194,-93 493,-138 801,-120 414,23 683,115 937,319 174,140 357,402 474,680 26,62 1945,5159 1945,5167 0,2 -232,2 -516,1 l -517,-3 -556,-1550 c -485,-1350 -560,-1550 -578,-1553 -18,-3 -26,11 -62,105 -23,59 -295,758 -605,1553 l -562,1445 -567,3 c -312,1 -567,0 -567,-2 z"
|
||||
/>
|
||||
<path
|
||||
d="m 15690,10867 c 0,-1970 -1,-1936 56,-2153 128,-497 461,-787 1014,-885 147,-26 440,-36 605,-20 413,40 834,200 1181,447 35,25 73,44 88,44 14,0 26,-1 26,-3 0,-2 20,-94 45,-205 25,-111 45,-204 45,-207 0,-3 209,-5 465,-5 h 465 v 2400 2400 h -535 -535 l -2,-1866 -3,-1866 -115,-50 c -425,-185 -743,-252 -1088,-227 -302,21 -472,109 -562,293 -79,161 -74,11 -77,1969 l -3,1747 h -535 -535 z"
|
||||
/>
|
||||
<path
|
||||
d="m 24785,12668 c -3,-7 -4,-1086 -3,-2398 l 3,-2385 538,-3 537,-2 v 2400 2400 h -535 c -419,0 -537,-3 -540,-12 z"
|
||||
/>
|
||||
<path
|
||||
d="m 21660,8275 v -545 h 565 565 v 545 545 h -565 -565 z"
|
||||
/>
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 4.0 KiB |
126
lexicons/com/atproto/repo/applyWrites.json
Normal file
126
lexicons/com/atproto/repo/applyWrites.json
Normal file
@@ -0,0 +1,126 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.applyWrites",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "procedure",
|
||||
"description": "Apply a batch transaction of repository creates, updates, and deletes. Requires auth, implemented by PDS.",
|
||||
"input": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["repo", "writes"],
|
||||
"properties": {
|
||||
"repo": {
|
||||
"type": "string",
|
||||
"format": "at-identifier",
|
||||
"description": "The handle or DID of the repo (aka, current account)."
|
||||
},
|
||||
"validate": {
|
||||
"type": "boolean",
|
||||
"description": "Can be set to 'false' to skip Lexicon schema validation of record data across all operations, 'true' to require it, or leave unset to validate only for known Lexicons."
|
||||
},
|
||||
"writes": {
|
||||
"type": "array",
|
||||
"items": {
|
||||
"type": "union",
|
||||
"refs": ["#create", "#update", "#delete"],
|
||||
"closed": true
|
||||
}
|
||||
},
|
||||
"swapCommit": {
|
||||
"type": "string",
|
||||
"description": "If provided, the entire operation will fail if the current repo commit CID does not match this value. Used to prevent conflicting repo mutations.",
|
||||
"format": "cid"
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": [],
|
||||
"properties": {
|
||||
"commit": {
|
||||
"type": "ref",
|
||||
"ref": "com.atproto.repo.defs#commitMeta"
|
||||
},
|
||||
"results": {
|
||||
"type": "array",
|
||||
"items": {
|
||||
"type": "union",
|
||||
"refs": ["#createResult", "#updateResult", "#deleteResult"],
|
||||
"closed": true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"errors": [
|
||||
{
|
||||
"name": "InvalidSwap",
|
||||
"description": "Indicates that the 'swapCommit' parameter did not match current commit."
|
||||
}
|
||||
]
|
||||
},
|
||||
"create": {
|
||||
"type": "object",
|
||||
"description": "Operation which creates a new record.",
|
||||
"required": ["collection", "value"],
|
||||
"properties": {
|
||||
"collection": { "type": "string", "format": "nsid" },
|
||||
"rkey": { "type": "string", "maxLength": 15 },
|
||||
"value": { "type": "unknown" }
|
||||
}
|
||||
},
|
||||
"update": {
|
||||
"type": "object",
|
||||
"description": "Operation which updates an existing record.",
|
||||
"required": ["collection", "rkey", "value"],
|
||||
"properties": {
|
||||
"collection": { "type": "string", "format": "nsid" },
|
||||
"rkey": { "type": "string" },
|
||||
"value": { "type": "unknown" }
|
||||
}
|
||||
},
|
||||
"delete": {
|
||||
"type": "object",
|
||||
"description": "Operation which deletes an existing record.",
|
||||
"required": ["collection", "rkey"],
|
||||
"properties": {
|
||||
"collection": { "type": "string", "format": "nsid" },
|
||||
"rkey": { "type": "string" }
|
||||
}
|
||||
},
|
||||
"createResult": {
|
||||
"type": "object",
|
||||
"required": ["uri", "cid"],
|
||||
"properties": {
|
||||
"uri": { "type": "string", "format": "at-uri" },
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"validationStatus": {
|
||||
"type": "string",
|
||||
"knownValues": ["valid", "unknown"]
|
||||
}
|
||||
}
|
||||
},
|
||||
"updateResult": {
|
||||
"type": "object",
|
||||
"required": ["uri", "cid"],
|
||||
"properties": {
|
||||
"uri": { "type": "string", "format": "at-uri" },
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"validationStatus": {
|
||||
"type": "string",
|
||||
"knownValues": ["valid", "unknown"]
|
||||
}
|
||||
}
|
||||
},
|
||||
"deleteResult": {
|
||||
"type": "object",
|
||||
"required": [],
|
||||
"properties": {}
|
||||
}
|
||||
}
|
||||
}
|
72
lexicons/com/atproto/repo/createRecord.json
Normal file
72
lexicons/com/atproto/repo/createRecord.json
Normal file
@@ -0,0 +1,72 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.createRecord",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "procedure",
|
||||
"description": "Create a single new repository record. Requires auth, implemented by PDS.",
|
||||
"input": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["repo", "collection", "record"],
|
||||
"properties": {
|
||||
"repo": {
|
||||
"type": "string",
|
||||
"format": "at-identifier",
|
||||
"description": "The handle or DID of the repo (aka, current account)."
|
||||
},
|
||||
"collection": {
|
||||
"type": "string",
|
||||
"format": "nsid",
|
||||
"description": "The NSID of the record collection."
|
||||
},
|
||||
"rkey": {
|
||||
"type": "string",
|
||||
"description": "The Record Key.",
|
||||
"maxLength": 15
|
||||
},
|
||||
"validate": {
|
||||
"type": "boolean",
|
||||
"description": "Can be set to 'false' to skip Lexicon schema validation of record data, 'true' to require it, or leave unset to validate only for known Lexicons."
|
||||
},
|
||||
"record": {
|
||||
"type": "unknown",
|
||||
"description": "The record itself. Must contain a $type field."
|
||||
},
|
||||
"swapCommit": {
|
||||
"type": "string",
|
||||
"format": "cid",
|
||||
"description": "Compare and swap with the previous commit by CID."
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["uri", "cid"],
|
||||
"properties": {
|
||||
"uri": { "type": "string", "format": "at-uri" },
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"commit": {
|
||||
"type": "ref",
|
||||
"ref": "com.atproto.repo.defs#commitMeta"
|
||||
},
|
||||
"validationStatus": {
|
||||
"type": "string",
|
||||
"knownValues": ["valid", "unknown"]
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"errors": [
|
||||
{
|
||||
"name": "InvalidSwap",
|
||||
"description": "Indicates that 'swapCommit' didn't match current repo commit."
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
14
lexicons/com/atproto/repo/defs.json
Normal file
14
lexicons/com/atproto/repo/defs.json
Normal file
@@ -0,0 +1,14 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.defs",
|
||||
"defs": {
|
||||
"commitMeta": {
|
||||
"type": "object",
|
||||
"required": ["cid", "rev"],
|
||||
"properties": {
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"rev": { "type": "string" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
56
lexicons/com/atproto/repo/deleteRecord.json
Normal file
56
lexicons/com/atproto/repo/deleteRecord.json
Normal file
@@ -0,0 +1,56 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.deleteRecord",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "procedure",
|
||||
"description": "Delete a repository record, or ensure it doesn't exist. Requires auth, implemented by PDS.",
|
||||
"input": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["repo", "collection", "rkey"],
|
||||
"properties": {
|
||||
"repo": {
|
||||
"type": "string",
|
||||
"format": "at-identifier",
|
||||
"description": "The handle or DID of the repo (aka, current account)."
|
||||
},
|
||||
"collection": {
|
||||
"type": "string",
|
||||
"format": "nsid",
|
||||
"description": "The NSID of the record collection."
|
||||
},
|
||||
"rkey": {
|
||||
"type": "string",
|
||||
"description": "The Record Key."
|
||||
},
|
||||
"swapRecord": {
|
||||
"type": "string",
|
||||
"format": "cid",
|
||||
"description": "Compare and swap with the previous record by CID."
|
||||
},
|
||||
"swapCommit": {
|
||||
"type": "string",
|
||||
"format": "cid",
|
||||
"description": "Compare and swap with the previous commit by CID."
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"commit": {
|
||||
"type": "ref",
|
||||
"ref": "com.atproto.repo.defs#commitMeta"
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"errors": [{ "name": "InvalidSwap" }]
|
||||
}
|
||||
}
|
||||
}
|
51
lexicons/com/atproto/repo/describeRepo.json
Normal file
51
lexicons/com/atproto/repo/describeRepo.json
Normal file
@@ -0,0 +1,51 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.describeRepo",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "query",
|
||||
"description": "Get information about an account and repository, including the list of collections. Does not require auth.",
|
||||
"parameters": {
|
||||
"type": "params",
|
||||
"required": ["repo"],
|
||||
"properties": {
|
||||
"repo": {
|
||||
"type": "string",
|
||||
"format": "at-identifier",
|
||||
"description": "The handle or DID of the repo."
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": [
|
||||
"handle",
|
||||
"did",
|
||||
"didDoc",
|
||||
"collections",
|
||||
"handleIsCorrect"
|
||||
],
|
||||
"properties": {
|
||||
"handle": { "type": "string", "format": "handle" },
|
||||
"did": { "type": "string", "format": "did" },
|
||||
"didDoc": {
|
||||
"type": "unknown",
|
||||
"description": "The complete DID document for this account."
|
||||
},
|
||||
"collections": {
|
||||
"type": "array",
|
||||
"description": "List of all the collections (NSIDs) for which this repo contains at least one record.",
|
||||
"items": { "type": "string", "format": "nsid" }
|
||||
},
|
||||
"handleIsCorrect": {
|
||||
"type": "boolean",
|
||||
"description": "Indicates if handle is currently valid (resolves bi-directionally)"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
45
lexicons/com/atproto/repo/getRecord.json
Normal file
45
lexicons/com/atproto/repo/getRecord.json
Normal file
@@ -0,0 +1,45 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.getRecord",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "query",
|
||||
"description": "Get a single record from a repository. Does not require auth.",
|
||||
"parameters": {
|
||||
"type": "params",
|
||||
"required": ["repo", "collection", "rkey"],
|
||||
"properties": {
|
||||
"repo": {
|
||||
"type": "string",
|
||||
"format": "at-identifier",
|
||||
"description": "The handle or DID of the repo."
|
||||
},
|
||||
"collection": {
|
||||
"type": "string",
|
||||
"format": "nsid",
|
||||
"description": "The NSID of the record collection."
|
||||
},
|
||||
"rkey": { "type": "string", "description": "The Record Key." },
|
||||
"cid": {
|
||||
"type": "string",
|
||||
"format": "cid",
|
||||
"description": "The CID of the version of the record. If not specified, then return the most recent version."
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["uri", "value"],
|
||||
"properties": {
|
||||
"uri": { "type": "string", "format": "at-uri" },
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"value": { "type": "unknown" }
|
||||
}
|
||||
}
|
||||
},
|
||||
"errors": [{ "name": "RecordNotFound" }]
|
||||
}
|
||||
}
|
||||
}
|
13
lexicons/com/atproto/repo/importRepo.json
Normal file
13
lexicons/com/atproto/repo/importRepo.json
Normal file
@@ -0,0 +1,13 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.importRepo",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "procedure",
|
||||
"description": "Import a repo in the form of a CAR file. Requires Content-Length HTTP header to be set.",
|
||||
"input": {
|
||||
"encoding": "application/vnd.ipld.car"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
44
lexicons/com/atproto/repo/listMissingBlobs.json
Normal file
44
lexicons/com/atproto/repo/listMissingBlobs.json
Normal file
@@ -0,0 +1,44 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.listMissingBlobs",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "query",
|
||||
"description": "Returns a list of missing blobs for the requesting account. Intended to be used in the account migration flow.",
|
||||
"parameters": {
|
||||
"type": "params",
|
||||
"properties": {
|
||||
"limit": {
|
||||
"type": "integer",
|
||||
"minimum": 1,
|
||||
"maximum": 1000,
|
||||
"default": 500
|
||||
},
|
||||
"cursor": { "type": "string" }
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["blobs"],
|
||||
"properties": {
|
||||
"cursor": { "type": "string" },
|
||||
"blobs": {
|
||||
"type": "array",
|
||||
"items": { "type": "ref", "ref": "#recordBlob" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"recordBlob": {
|
||||
"type": "object",
|
||||
"required": ["cid", "recordUri"],
|
||||
"properties": {
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"recordUri": { "type": "string", "format": "at-uri" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
69
lexicons/com/atproto/repo/listRecords.json
Normal file
69
lexicons/com/atproto/repo/listRecords.json
Normal file
@@ -0,0 +1,69 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.listRecords",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "query",
|
||||
"description": "List a range of records in a repository, matching a specific collection. Does not require auth.",
|
||||
"parameters": {
|
||||
"type": "params",
|
||||
"required": ["repo", "collection"],
|
||||
"properties": {
|
||||
"repo": {
|
||||
"type": "string",
|
||||
"format": "at-identifier",
|
||||
"description": "The handle or DID of the repo."
|
||||
},
|
||||
"collection": {
|
||||
"type": "string",
|
||||
"format": "nsid",
|
||||
"description": "The NSID of the record type."
|
||||
},
|
||||
"limit": {
|
||||
"type": "integer",
|
||||
"minimum": 1,
|
||||
"maximum": 100,
|
||||
"default": 50,
|
||||
"description": "The number of records to return."
|
||||
},
|
||||
"cursor": { "type": "string" },
|
||||
"rkeyStart": {
|
||||
"type": "string",
|
||||
"description": "DEPRECATED: The lowest sort-ordered rkey to start from (exclusive)"
|
||||
},
|
||||
"rkeyEnd": {
|
||||
"type": "string",
|
||||
"description": "DEPRECATED: The highest sort-ordered rkey to stop at (exclusive)"
|
||||
},
|
||||
"reverse": {
|
||||
"type": "boolean",
|
||||
"description": "Flag to reverse the order of the returned records."
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["records"],
|
||||
"properties": {
|
||||
"cursor": { "type": "string" },
|
||||
"records": {
|
||||
"type": "array",
|
||||
"items": { "type": "ref", "ref": "#record" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"record": {
|
||||
"type": "object",
|
||||
"required": ["uri", "cid", "value"],
|
||||
"properties": {
|
||||
"uri": { "type": "string", "format": "at-uri" },
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"value": { "type": "unknown" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
73
lexicons/com/atproto/repo/putRecord.json
Normal file
73
lexicons/com/atproto/repo/putRecord.json
Normal file
@@ -0,0 +1,73 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.putRecord",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "procedure",
|
||||
"description": "Write a repository record, creating or updating it as needed. Requires auth, implemented by PDS.",
|
||||
"input": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["repo", "collection", "rkey", "record"],
|
||||
"nullable": ["swapRecord"],
|
||||
"properties": {
|
||||
"repo": {
|
||||
"type": "string",
|
||||
"format": "at-identifier",
|
||||
"description": "The handle or DID of the repo (aka, current account)."
|
||||
},
|
||||
"collection": {
|
||||
"type": "string",
|
||||
"format": "nsid",
|
||||
"description": "The NSID of the record collection."
|
||||
},
|
||||
"rkey": {
|
||||
"type": "string",
|
||||
"description": "The Record Key.",
|
||||
"maxLength": 15
|
||||
},
|
||||
"validate": {
|
||||
"type": "boolean",
|
||||
"description": "Can be set to 'false' to skip Lexicon schema validation of record data, 'true' to require it, or leave unset to validate only for known Lexicons."
|
||||
},
|
||||
"record": {
|
||||
"type": "unknown",
|
||||
"description": "The record to write."
|
||||
},
|
||||
"swapRecord": {
|
||||
"type": "string",
|
||||
"format": "cid",
|
||||
"description": "Compare and swap with the previous record by CID. WARNING: nullable and optional field; may cause problems with golang implementation"
|
||||
},
|
||||
"swapCommit": {
|
||||
"type": "string",
|
||||
"format": "cid",
|
||||
"description": "Compare and swap with the previous commit by CID."
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["uri", "cid"],
|
||||
"properties": {
|
||||
"uri": { "type": "string", "format": "at-uri" },
|
||||
"cid": { "type": "string", "format": "cid" },
|
||||
"commit": {
|
||||
"type": "ref",
|
||||
"ref": "com.atproto.repo.defs#commitMeta"
|
||||
},
|
||||
"validationStatus": {
|
||||
"type": "string",
|
||||
"knownValues": ["valid", "unknown"]
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"errors": [{ "name": "InvalidSwap" }]
|
||||
}
|
||||
}
|
||||
}
|
15
lexicons/com/atproto/repo/strongRef.json
Normal file
15
lexicons/com/atproto/repo/strongRef.json
Normal file
@@ -0,0 +1,15 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.strongRef",
|
||||
"description": "A URI with a content-hash fingerprint.",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "object",
|
||||
"required": ["uri", "cid"],
|
||||
"properties": {
|
||||
"uri": { "type": "string", "format": "at-uri" },
|
||||
"cid": { "type": "string", "format": "cid" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
23
lexicons/com/atproto/repo/uploadBlob.json
Normal file
23
lexicons/com/atproto/repo/uploadBlob.json
Normal file
@@ -0,0 +1,23 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "com.atproto.repo.uploadBlob",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "procedure",
|
||||
"description": "Upload a new blob, to be referenced from a repository record. The blob will be deleted if it is not referenced within a time window (eg, minutes). Blob restrictions (mimetype, size, etc) are enforced when the reference is created. Requires auth, implemented by PDS.",
|
||||
"input": {
|
||||
"encoding": "*/*"
|
||||
},
|
||||
"output": {
|
||||
"encoding": "application/json",
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"required": ["blob"],
|
||||
"properties": {
|
||||
"blob": { "type": "blob" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
0
repos/.keep
Normal file
0
repos/.keep
Normal file
1
repos/atproto
Submodule
1
repos/atproto
Submodule
Submodule repos/atproto added at 8f2b80a0dc
1
repos/bluesky-selfhost-env
Submodule
1
repos/bluesky-selfhost-env
Submodule
Submodule repos/bluesky-selfhost-env added at 3e384e3b58
1
repos/cookbook
Submodule
1
repos/cookbook
Submodule
Submodule repos/cookbook added at 03739f5268
1
repos/did-method-plc
Submodule
1
repos/did-method-plc
Submodule
Submodule repos/did-method-plc added at 13da315787
1
repos/feed-generator
Submodule
1
repos/feed-generator
Submodule
Submodule repos/feed-generator added at 9a887dd8f2
1
repos/social-app
Submodule
1
repos/social-app
Submodule
Submodule repos/social-app added at f6649e22a7
0
repos_extra/.keep
Normal file
0
repos_extra/.keep
Normal file
499
repos_extra/frontpage/diff.patch
Normal file
499
repos_extra/frontpage/diff.patch
Normal file
@@ -0,0 +1,499 @@
|
||||
diff --git a/packages-rs/drainpipe/docker-compose.yml b/packages-rs/drainpipe/docker-compose.yml
|
||||
index 0e1d2ca..d5e214a 100644
|
||||
--- a/packages-rs/drainpipe/docker-compose.yml
|
||||
+++ b/packages-rs/drainpipe/docker-compose.yml
|
||||
@@ -1,4 +1,3 @@
|
||||
-version: "3"
|
||||
services:
|
||||
drainpipe:
|
||||
build:
|
||||
diff --git a/packages-rs/drainpipe/src/main.rs b/packages-rs/drainpipe/src/main.rs
|
||||
index 21278a5..0a05d06 100644
|
||||
--- a/packages-rs/drainpipe/src/main.rs
|
||||
+++ b/packages-rs/drainpipe/src/main.rs
|
||||
@@ -76,7 +76,7 @@ async fn process(message: Vec<u8>, ctx: &mut Context) -> Result<i64, ProcessErro
|
||||
let frontpage_ops = commit
|
||||
.operations
|
||||
.iter()
|
||||
- .filter(|op| op.path.starts_with("fyi.unravel.frontpage."))
|
||||
+ .filter(|op| op.path.starts_with("ai.syui.o."))
|
||||
.collect::<Vec<_>>();
|
||||
if !frontpage_ops.is_empty() {
|
||||
process_frontpage_ops(&frontpage_ops, &commit, &ctx)
|
||||
diff --git a/packages/frontpage/app/(app)/_components/post-card.tsx b/packages/frontpage/app/(app)/_components/post-card.tsx
|
||||
index c0720e5..e619a5f 100644
|
||||
--- a/packages/frontpage/app/(app)/_components/post-card.tsx
|
||||
+++ b/packages/frontpage/app/(app)/_components/post-card.tsx
|
||||
@@ -83,7 +83,7 @@ export async function PostCard({
|
||||
votes={votes}
|
||||
/>
|
||||
</div>
|
||||
- <div className="w-full">
|
||||
+ <div className="w-full overflow-hidden">
|
||||
<h2 className="mb-1 text-xl">
|
||||
<a
|
||||
href={url}
|
||||
diff --git a/packages/frontpage/app/(app)/layout.tsx b/packages/frontpage/app/(app)/layout.tsx
|
||||
index 232502d..8340a9f 100644
|
||||
--- a/packages/frontpage/app/(app)/layout.tsx
|
||||
+++ b/packages/frontpage/app/(app)/layout.tsx
|
||||
@@ -31,7 +31,8 @@ export default async function Layout({
|
||||
}) {
|
||||
const session = await getSession();
|
||||
return (
|
||||
- <div className="container mx-auto px-4 md:px-6 pt-4 pb-8 md:py-12 max-w-3xl">
|
||||
+ <div className="container mx-auto px-4 md:px-6 pt-4 pb-8 md:py-12">
|
||||
+ {/* <div className="container mx-auto px-4 md:px-6 pt-4 pb-8 md:py-12 max-w-3xl"> */}
|
||||
<div className="flex place-content-between items-center mb-8">
|
||||
<Link href="/">
|
||||
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||
@@ -52,6 +53,8 @@ export default async function Layout({
|
||||
|
||||
<main className="mb-6">{children}</main>
|
||||
|
||||
+ <footer> <span className="flex flex-row-reverse"><a href="https://github.com/likeandscribe/frontpage/" className="text-blue-600"><img src="https://raw.githubusercontent.com/likeandscribe/frontpage/refs/heads/main/packages/frontpage/public/frontpage-logo.svg" alt="Frontpage | MIT License | https://github.com/likeandscribe/frontpage/blob/main/LICENSE" className="h-10" /></a></span> <span className="flex justify-center items-center text-gray-500 dark:text-gray-400"> <a href={`https://bsky.app/profile/${FRONTPAGE_ATPROTO_HANDLE}`} className="font-medium text-indigo-600 hover:text-indigo-500 dark:text-indigo-400 dark:hover:text-indigo-300" > @syui.ai </a> </span> </footer>
|
||||
+ {/*
|
||||
<footer className="flex justify-between items-center text-gray-500 dark:text-gray-400">
|
||||
<p>
|
||||
Made by{" "}
|
||||
@@ -59,10 +62,11 @@ export default async function Layout({
|
||||
href={`https://bsky.app/profile/${FRONTPAGE_ATPROTO_HANDLE}`}
|
||||
className="font-medium text-indigo-600 hover:text-indigo-500 dark:text-indigo-400 dark:hover:text-indigo-300"
|
||||
>
|
||||
- @frontpage.fyi <OpenInNewWindowIcon className="inline" />
|
||||
+ @o.syui.ai <OpenInNewWindowIcon className="inline" />
|
||||
</a>
|
||||
</p>
|
||||
</footer>
|
||||
+ */}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -141,8 +145,8 @@ async function LoginOrLogout() {
|
||||
}
|
||||
|
||||
return (
|
||||
- <Button variant="outline" asChild>
|
||||
- <Link href="/login">Login</Link>
|
||||
+ <Button asChild>
|
||||
+ <Link href="/login">@</Link>
|
||||
</Button>
|
||||
);
|
||||
}
|
||||
diff --git a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/[commentAuthor]/[commentRkey]/page.tsx b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/[commentAuthor]/[commentRkey]/page.tsx
|
||||
index c9a2883..c50b02f 100644
|
||||
--- a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/[commentAuthor]/[commentRkey]/page.tsx
|
||||
+++ b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/[commentAuthor]/[commentRkey]/page.tsx
|
||||
@@ -30,7 +30,7 @@ export async function generateMetadata(props: {
|
||||
description:
|
||||
comment.status === "live" ? truncateText(comment.body, 47) : null,
|
||||
alternates: {
|
||||
- canonical: `https://frontpage.fyi${path}`,
|
||||
+ canonical: `https://o.syui.ai${path}`,
|
||||
},
|
||||
openGraph:
|
||||
comment.status === "live"
|
||||
@@ -40,7 +40,7 @@ export async function generateMetadata(props: {
|
||||
type: "article",
|
||||
publishedTime: comment.createdAt.toISOString(),
|
||||
authors: [`@${handle}`],
|
||||
- url: `https://frontpage.fyi${path}`,
|
||||
+ url: `https://o.syui.ai${path}`,
|
||||
images: [
|
||||
{
|
||||
url: `${path}/og-image`,
|
||||
diff --git a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx
|
||||
index 71dd664..954f499 100644
|
||||
--- a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx
|
||||
+++ b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx
|
||||
@@ -283,16 +283,12 @@ export function NewComment({
|
||||
ref={textAreaRef}
|
||||
placeholder="Write a comment..."
|
||||
disabled={isPending}
|
||||
- className="resize-y flex-1"
|
||||
+ className=""
|
||||
/>
|
||||
<div className="w-full flex justify-between">
|
||||
- <InputLengthIndicator
|
||||
- length={input.length}
|
||||
- maxLength={MAX_COMMENT_LENGTH}
|
||||
- />
|
||||
- <div className="flex gap-2">
|
||||
+ <div className="w-full">
|
||||
{extraButton}
|
||||
- <Button type="submit" disabled={isPending}>
|
||||
+ <Button type="submit" disabled={isPending} className="w-full">
|
||||
{isPending ? <Spinner /> : "Post"}
|
||||
</Button>
|
||||
</div>
|
||||
diff --git a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx
|
||||
index 517c871..ef73ecb 100644
|
||||
--- a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx
|
||||
+++ b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx
|
||||
@@ -29,7 +29,7 @@ export default async function Post(props: {
|
||||
}
|
||||
|
||||
return (
|
||||
- <main className="mx-auto max-w-4xl space-y-6">
|
||||
+ <main className="mx-auto space-y-1">
|
||||
<PostCard
|
||||
author={post.authorDid}
|
||||
createdAt={post.createdAt}
|
||||
diff --git a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx
|
||||
index a83f810..11003f9 100644
|
||||
--- a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx
|
||||
+++ b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx
|
||||
@@ -20,7 +20,7 @@ export async function generateMetadata(props: {
|
||||
title: post.title,
|
||||
description: "Discuss this post on Frontpage.",
|
||||
alternates: {
|
||||
- canonical: `https://frontpage.fyi${path}`,
|
||||
+ canonical: `https://o.syui.ai${path}`,
|
||||
},
|
||||
openGraph: {
|
||||
title: post.title,
|
||||
@@ -28,7 +28,7 @@ export async function generateMetadata(props: {
|
||||
type: "article",
|
||||
publishedTime: post.createdAt.toISOString(),
|
||||
authors: [`@${handle}`],
|
||||
- url: `https://frontpage.fyi${path}`,
|
||||
+ url: `https://o.syui.ai${path}`,
|
||||
images: [
|
||||
{
|
||||
url: `${path}/og-image`,
|
||||
@@ -45,6 +45,8 @@ export default async function Post(props: { params: Promise<PostPageParams> }) {
|
||||
|
||||
return (
|
||||
<>
|
||||
+
|
||||
+ {/*
|
||||
<LinkAlternateAtUri
|
||||
authority={authorDid}
|
||||
collection={PostCollection}
|
||||
@@ -65,6 +67,71 @@ export default async function Post(props: { params: Promise<PostPageParams> }) {
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
+ */}
|
||||
+
|
||||
+ {params.postAuthor === "syui.ai" && (post.title.includes("live") || post.title.includes("video")) && (
|
||||
+ <div className="flex py-10 max-h-[720px] md:h-screen space-x-8">
|
||||
+ <iframe className="w-full h-[300px] sm:h-[450px] md:h-[630px]
|
||||
+ [&::-webkit-scrollbar]:w-2
|
||||
+ [&::-webkit-scrollbar-track]:bg-gray-100
|
||||
+ [&::-webkit-scrollbar-thumb]:bg-gray-300
|
||||
+ dark:[&::-webkit-scrollbar-track]:bg-neutral-700
|
||||
+ dark:[&::-webkit-scrollbar-thumb]:bg-neutral-500"
|
||||
+ title="stream" src={post.url}>
|
||||
+ </iframe>
|
||||
+ <div className="hidden w-64 hidden-scrollbar md:block md:overflow-y-scroll
|
||||
+ max-h-[600px]
|
||||
+ [&::-webkit-scrollbar]:w-2
|
||||
+ [&::-webkit-scrollbar-track]:bg-gray-100
|
||||
+ [&::-webkit-scrollbar-thumb]:bg-gray-300
|
||||
+ dark:[&::-webkit-scrollbar-track]:bg-neutral-700
|
||||
+ dark:[&::-webkit-scrollbar-thumb]:bg-neutral-500">
|
||||
+ <LinkAlternateAtUri
|
||||
+ authority={authorDid}
|
||||
+ collection={PostCollection}
|
||||
+ rkey={post.rkey}
|
||||
+ />
|
||||
+ <NewComment postRkey={post.rkey} postAuthorDid={authorDid} />
|
||||
+ {comments.map((comment) => (
|
||||
+ <Comment
|
||||
+ key={comment.id}
|
||||
+ comment={comment}
|
||||
+ level={0}
|
||||
+ postAuthorParam={params.postAuthor}
|
||||
+ postRkey={post.rkey}
|
||||
+ />
|
||||
+ ))}
|
||||
+ </div>
|
||||
+ </div>
|
||||
+ )}
|
||||
+ <LinkAlternateAtUri
|
||||
+ authority={authorDid}
|
||||
+ collection={PostCollection}
|
||||
+ rkey={post.rkey}
|
||||
+ />
|
||||
+ {post.status === "live" ? (
|
||||
+ <NewComment postRkey={post.rkey} postAuthorDid={authorDid} />
|
||||
+ ) : (
|
||||
+ <Alert>
|
||||
+ <AlertTitle>This post has been deleted</AlertTitle>
|
||||
+ <AlertDescription>
|
||||
+ Deleted posts cannot receive new comments.
|
||||
+ </AlertDescription>
|
||||
+ </Alert>
|
||||
+ )}
|
||||
+ <div className="flex flex-col gap-6 bg-white">
|
||||
+ {comments.map((comment) => (
|
||||
+ <Comment
|
||||
+ key={comment.id}
|
||||
+ comment={comment}
|
||||
+ level={0}
|
||||
+ postAuthorParam={params.postAuthor}
|
||||
+ postRkey={post.rkey}
|
||||
+ />
|
||||
+ ))}
|
||||
+
|
||||
+ </div>
|
||||
+
|
||||
</>
|
||||
);
|
||||
}
|
||||
diff --git a/packages/frontpage/app/(auth)/login/page.tsx b/packages/frontpage/app/(auth)/login/page.tsx
|
||||
index e3165f9..4d661ce 100644
|
||||
--- a/packages/frontpage/app/(auth)/login/page.tsx
|
||||
+++ b/packages/frontpage/app/(auth)/login/page.tsx
|
||||
@@ -14,17 +14,15 @@ export default async function LoginPage() {
|
||||
<div className="w-full max-w-md space-y-6">
|
||||
<div className="text-center">
|
||||
<h2 className="text-3xl font-bold tracking-tight text-gray-900 dark:text-gray-100">
|
||||
- Sign in to Frontpage
|
||||
+ Login with <span className="text-blue-600">at</span>proto
|
||||
</h2>
|
||||
<p className="mt-2 text-sm text-gray-600 dark:text-gray-400">
|
||||
- Don't have an account?{" "}
|
||||
<a
|
||||
href="https://bsky.app/"
|
||||
className="font-medium text-indigo-600 hover:text-indigo-500 dark:text-indigo-400 dark:hover:text-indigo-300"
|
||||
>
|
||||
- Sign up on Bluesky
|
||||
- </a>
|
||||
- , then return here to login.
|
||||
+ Bluesky
|
||||
+ </a> oauth login
|
||||
</p>
|
||||
</div>
|
||||
<LoginForm />
|
||||
diff --git a/packages/frontpage/app/api/receive_hook/route.ts b/packages/frontpage/app/api/receive_hook/route.ts
|
||||
index 04bda27..9feb136 100644
|
||||
--- a/packages/frontpage/app/api/receive_hook/route.ts
|
||||
+++ b/packages/frontpage/app/api/receive_hook/route.ts
|
||||
@@ -96,7 +96,7 @@ export async function POST(request: Request) {
|
||||
});
|
||||
}
|
||||
|
||||
- if (collection === "fyi.unravel.frontpage.vote") {
|
||||
+ if (collection === "ai.syui.o.vote") {
|
||||
if (op.action === "create") {
|
||||
const hydratedRecord = await atprotoGetRecord({
|
||||
serviceEndpoint: service,
|
||||
diff --git a/packages/frontpage/app/globals.css b/packages/frontpage/app/globals.css
|
||||
index 9abc86b..e22dba5 100644
|
||||
--- a/packages/frontpage/app/globals.css
|
||||
+++ b/packages/frontpage/app/globals.css
|
||||
@@ -13,8 +13,12 @@
|
||||
--popover: 0 0% 100%;
|
||||
--popover-foreground: 222.2 84% 4.9%;
|
||||
|
||||
+ /*
|
||||
--primary: 222.2 47.4% 11.2%;
|
||||
- --primary-foreground: 210 40% 98%;
|
||||
+ --primary-foreground: 210 40% 98%; */
|
||||
+
|
||||
+ --primary: 210 100% 50%; /* HSLカラーで青色を定義 */
|
||||
+ --primary-foreground: 0 0% 100%; /* 白色のテキスト */
|
||||
|
||||
--secondary: 210 40% 96.1%;
|
||||
--secondary-foreground: 222.2 47.4% 11.2%;
|
||||
@@ -67,8 +71,12 @@
|
||||
--popover: 222.2 84% 4.9%;
|
||||
--popover-foreground: 210 40% 98%;
|
||||
|
||||
- --primary: 210 40% 98%;
|
||||
- --primary-foreground: 222.2 47.4% 11.2%;
|
||||
+ /*--primary: 210 40% 98%;
|
||||
+ *--primary-foreground: 222.2 47.4% 11.2%;*/
|
||||
+
|
||||
+ /* ダークモード用に少し明るい青 */
|
||||
+ --primary: 210 100% 60%;
|
||||
+ --primary-foreground: 0 0% 100%;
|
||||
|
||||
--secondary: 217.2 32.6% 17.5%;
|
||||
--secondary-foreground: 210 40% 98%;
|
||||
diff --git a/packages/frontpage/app/layout.tsx b/packages/frontpage/app/layout.tsx
|
||||
index e8b5a39..a9fa945 100644
|
||||
--- a/packages/frontpage/app/layout.tsx
|
||||
+++ b/packages/frontpage/app/layout.tsx
|
||||
@@ -22,13 +22,13 @@ const sourceSans = Source_Sans_3({
|
||||
});
|
||||
|
||||
export const metadata: Metadata = {
|
||||
- title: "Frontpage",
|
||||
+ title: "o.syui.ai",
|
||||
description: "A federated link aggregator. Your frontpage to the internet.",
|
||||
openGraph: {
|
||||
title: "Frontpage",
|
||||
description: "Your frontpage to the internet.",
|
||||
type: "website",
|
||||
- siteName: "frontpage.fyi",
|
||||
+ siteName: "o.syui.ai",
|
||||
},
|
||||
};
|
||||
|
||||
@@ -53,11 +53,7 @@ export default function Layout({ children }: { children: React.ReactNode }) {
|
||||
<SpeedInsights />
|
||||
<Analytics />
|
||||
</ThemeProvider>
|
||||
- <script
|
||||
- defer
|
||||
- src="https://static.cloudflareinsights.com/beacon.min.js"
|
||||
- data-cf-beacon='{"token": "5e64b32c70b34b1583d863032b03d9ad"}'
|
||||
- />
|
||||
+ {/* <script defer src="https://static.cloudflareinsights.com/beacon.min.js" data-cf-beacon='{"token": "5e64b32c70b34b1583d863032b03d9ad"}' /> */}
|
||||
</body>
|
||||
</html>
|
||||
);
|
||||
diff --git a/packages/frontpage/lib/auth.ts b/packages/frontpage/lib/auth.ts
|
||||
index ba161a0..cadfe54 100644
|
||||
--- a/packages/frontpage/lib/auth.ts
|
||||
+++ b/packages/frontpage/lib/auth.ts
|
||||
@@ -33,7 +33,7 @@ import { eq } from "drizzle-orm";
|
||||
import { getDidFromHandleOrDid } from "./data/atproto/identity";
|
||||
import { startSpan } from "@sentry/nextjs";
|
||||
|
||||
-const USER_AGENT = "appview/@frontpage.fyi (@tom-sherman.com)";
|
||||
+const USER_AGENT = "appview/@o.syui.ai (@tom-sherman.com)";
|
||||
|
||||
export const getPrivateJwk = cache(() =>
|
||||
importJWK(JSON.parse(process.env.PRIVATE_JWK!), USER_SESSION_JWT_ALG),
|
||||
diff --git a/packages/frontpage/lib/constants.ts b/packages/frontpage/lib/constants.ts
|
||||
index 59b2de5..8fd68b5 100644
|
||||
--- a/packages/frontpage/lib/constants.ts
|
||||
+++ b/packages/frontpage/lib/constants.ts
|
||||
@@ -1 +1 @@
|
||||
-export const FRONTPAGE_ATPROTO_HANDLE = "frontpage.fyi";
|
||||
+export const FRONTPAGE_ATPROTO_HANDLE = "syui.ai";
|
||||
diff --git a/packages/frontpage/lib/data/atproto/comment.ts b/packages/frontpage/lib/data/atproto/comment.ts
|
||||
index 42dfb3b..ed0d6df 100644
|
||||
--- a/packages/frontpage/lib/data/atproto/comment.ts
|
||||
+++ b/packages/frontpage/lib/data/atproto/comment.ts
|
||||
@@ -11,7 +11,7 @@ import { PostCollection } from "./post";
|
||||
import { DID, getPdsUrl } from "./did";
|
||||
import { MAX_COMMENT_LENGTH } from "../db/constants";
|
||||
|
||||
-export const CommentCollection = "fyi.unravel.frontpage.comment";
|
||||
+export const CommentCollection = "ai.syui.o.comment";
|
||||
|
||||
export const CommentRecord = z.object({
|
||||
content: z.string().max(MAX_COMMENT_LENGTH),
|
||||
diff --git a/packages/frontpage/lib/data/atproto/event.ts b/packages/frontpage/lib/data/atproto/event.ts
|
||||
index 04fb4b0..80a9cb4 100644
|
||||
--- a/packages/frontpage/lib/data/atproto/event.ts
|
||||
+++ b/packages/frontpage/lib/data/atproto/event.ts
|
||||
@@ -9,7 +9,7 @@ import { isDid } from "./did";
|
||||
export const Collection = z.union([
|
||||
z.literal(PostCollection),
|
||||
z.literal(CommentCollection),
|
||||
- z.literal("fyi.unravel.frontpage.vote"),
|
||||
+ z.literal("ai.syui.o.vote"),
|
||||
]);
|
||||
|
||||
const Path = z.string().transform((p, ctx) => {
|
||||
diff --git a/packages/frontpage/lib/data/atproto/post.ts b/packages/frontpage/lib/data/atproto/post.ts
|
||||
index 8d8f930..eec4798 100644
|
||||
--- a/packages/frontpage/lib/data/atproto/post.ts
|
||||
+++ b/packages/frontpage/lib/data/atproto/post.ts
|
||||
@@ -9,7 +9,7 @@ import { DataLayerError } from "../error";
|
||||
import { DID, getPdsUrl } from "./did";
|
||||
import { MAX_POST_TITLE_LENGTH, MAX_POST_URL_LENGTH } from "../db/constants";
|
||||
|
||||
-export const PostCollection = "fyi.unravel.frontpage.post";
|
||||
+export const PostCollection = "ai.syui.o.post";
|
||||
|
||||
export const PostRecord = z.object({
|
||||
title: z.string().max(MAX_POST_TITLE_LENGTH),
|
||||
diff --git a/packages/frontpage/lib/data/atproto/vote.ts b/packages/frontpage/lib/data/atproto/vote.ts
|
||||
index 93bdb78..91a57c0 100644
|
||||
--- a/packages/frontpage/lib/data/atproto/vote.ts
|
||||
+++ b/packages/frontpage/lib/data/atproto/vote.ts
|
||||
@@ -49,7 +49,7 @@ export async function createVote({
|
||||
VoteRecord.parse(record);
|
||||
|
||||
await atprotoCreateRecord({
|
||||
- collection: "fyi.unravel.frontpage.vote",
|
||||
+ collection: "ai.syui.o.vote",
|
||||
record: record,
|
||||
});
|
||||
}
|
||||
@@ -58,7 +58,7 @@ export async function deleteVote(rkey: string) {
|
||||
await ensureUser();
|
||||
|
||||
await atprotoDeleteRecord({
|
||||
- collection: "fyi.unravel.frontpage.vote",
|
||||
+ collection: "ai.syui.o.vote",
|
||||
rkey,
|
||||
});
|
||||
}
|
||||
diff --git a/packages/frontpage/lib/data/db/post.ts b/packages/frontpage/lib/data/db/post.ts
|
||||
index 082d8af..4ba3a3c 100644
|
||||
--- a/packages/frontpage/lib/data/db/post.ts
|
||||
+++ b/packages/frontpage/lib/data/db/post.ts
|
||||
@@ -207,13 +207,13 @@ export async function unauthed_createPost({
|
||||
{
|
||||
title: "New post on Frontpage",
|
||||
description: post.title,
|
||||
- url: `https://frontpage.fyi/post/${authorDid}/${rkey}`,
|
||||
+ url: `https://o.syui.ai/post/${authorDid}/${rkey}`,
|
||||
color: 10181046,
|
||||
author: bskyProfile
|
||||
? {
|
||||
name: `@${bskyProfile.handle}`,
|
||||
icon_url: bskyProfile.avatar,
|
||||
- url: `https://frontpage.fyi/profile/${bskyProfile.handle}`,
|
||||
+ url: `https://o.syui.ai/profile/${bskyProfile.handle}`,
|
||||
}
|
||||
: undefined,
|
||||
fields: [
|
||||
diff --git a/packages/frontpage/public/frontpage-logo.svg b/packages/frontpage/public/frontpage-logo.svg
|
||||
index 191ff3a..bd39143 100644
|
||||
--- a/packages/frontpage/public/frontpage-logo.svg
|
||||
+++ b/packages/frontpage/public/frontpage-logo.svg
|
||||
@@ -1,30 +1,16 @@
|
||||
-<svg width="334" height="334" viewBox="0 0 334 334" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
-<style>
|
||||
- .stop-2 {
|
||||
- stop-color: #040818;
|
||||
- }
|
||||
-
|
||||
- @media (prefers-color-scheme: dark) {
|
||||
- .stop-2 {
|
||||
- stop-color: #fff;
|
||||
- }
|
||||
- }
|
||||
-</style>
|
||||
-<path d="M95 225.903V101L162 62.0968V32L69 86V241L95 225.903Z" fill="url(#paint0_linear_8_11)" style=""/>
|
||||
-<path d="M147 256.903V132L214 93.0968V63L121 117V272L147 256.903Z" fill="url(#paint1_linear_8_11)" style=""/>
|
||||
-<path d="M266 93V129L204 165V198L266 162V198L204 234V284L173 302V147L266 93Z" fill="url(#paint2_linear_8_11)" style=""/>
|
||||
-<defs>
|
||||
-<linearGradient id="paint0_linear_8_11" x1="69" y1="84.5" x2="205.5" y2="167" gradientUnits="userSpaceOnUse">
|
||||
- <stop stop-color="#2E05FF"/>
|
||||
- <stop class="stop-2" offset="1" />
|
||||
-</linearGradient>
|
||||
-<linearGradient id="paint1_linear_8_11" x1="69" y1="84.5" x2="205.5" y2="167" gradientUnits="userSpaceOnUse">
|
||||
- <stop stop-color="#2E05FF"/>
|
||||
- <stop class="stop-2" offset="1" />
|
||||
-</linearGradient>
|
||||
-<linearGradient id="paint2_linear_8_11" x1="69" y1="84.5" x2="205.5" y2="167" gradientUnits="userSpaceOnUse">
|
||||
- <stop stop-color="#2E05FF"/>
|
||||
- <stop class="stop-2" offset="1" />
|
||||
-</linearGradient>
|
||||
-</defs>
|
||||
+<?xml version="1.0" encoding="UTF-8"?>
|
||||
+<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||
+<svg xmlns="http://www.w3.org/2000/svg" version="1.1" width="1417px" height="1417px" style="shape-rendering:geometricPrecision; text-rendering:geometricPrecision; image-rendering:optimizeQuality; fill-rule:evenodd; clip-rule:evenodd" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||
+<g><path style="opacity:1" fill="#303030" d="M 717.5,182.5 C 716.773,186.224 718.106,188.891 721.5,190.5C 729.374,205.903 738.541,220.569 749,234.5C 750.399,237.363 751.399,240.363 752,243.5C 755.595,248.061 759.095,252.728 762.5,257.5C 763.792,259.667 764.625,262 765,264.5C 766.697,265.957 768.197,267.623 769.5,269.5C 773.055,277.685 777.722,285.018 783.5,291.5C 790.166,305.134 798.166,318.134 807.5,330.5C 814.742,342.185 821.742,353.852 828.5,365.5C 844.594,370.003 860.427,375.336 876,381.5C 877.169,382.009 878.002,382.842 878.5,384C 882.957,384.143 886.623,385.81 889.5,389C 891.715,390.044 894.048,390.71 896.5,391C 901.981,393.275 906.981,396.275 911.5,400C 913.518,400.505 915.518,401.005 917.5,401.5C 917.414,402.496 917.748,403.329 918.5,404C 923.387,405.947 928.054,408.114 932.5,410.5C 943.583,419.552 955.25,427.885 967.5,435.5C 979.922,445.75 991.922,456.583 1003.5,468C 1004.5,468.333 1005.5,468.667 1006.5,469C 1007.1,472.436 1009.1,474.269 1012.5,474.5C 1015.38,480.22 1019.55,484.887 1025,488.5C 1025.33,489.5 1025.67,490.5 1026,491.5C 1033.43,499.992 1040.6,508.658 1047.5,517.5C 1123.83,622.989 1147.5,739.989 1118.5,868.5C 1118.67,870.167 1118.83,871.833 1119,873.5C 1121.94,879.062 1125.11,884.395 1128.5,889.5C 1157.69,942.876 1187.19,996.21 1217,1049.5C 1218.4,1050.73 1219.57,1052.06 1220.5,1053.5C 1221.61,1056.18 1221.61,1058.84 1220.5,1061.5C 1147.34,1066.45 1074,1069.95 1000.5,1072C 988.536,1079.45 977.536,1088.28 967.5,1098.5C 956.996,1104.65 946.996,1111.65 937.5,1119.5C 930.045,1122.31 923.378,1126.31 917.5,1131.5C 911.537,1133.48 905.87,1135.98 900.5,1139C 898.015,1142.24 894.681,1143.91 890.5,1144C 889.833,1145.33 888.833,1146.33 887.5,1147C 855.91,1161.47 822.91,1171.64 788.5,1177.5C 782.486,1176.22 776.486,1176.39 770.5,1178C 772.5,1178.33 774.5,1178.67 776.5,1179C 774.618,1179.2 772.952,1179.7 771.5,1180.5C 674.89,1194.18 583.89,1177.51 498.5,1130.5C 493.758,1126.48 488.758,1122.98 483.5,1120C 479.247,1119.45 475.914,1117.45 473.5,1114C 461.022,1107.13 449.689,1098.8 439.5,1089C 438.5,1088.67 437.5,1088.33 436.5,1088C 432.5,1084.06 428.5,1080.06 424.5,1076C 421.495,1074.83 418.495,1073.66 415.5,1072.5C 360.479,1070.72 305.479,1068.22 250.5,1065C 236.756,1064.54 223.089,1063.38 209.5,1061.5C 197.677,1066.34 193.843,1063.01 198,1051.5C 201.275,1047.94 203.775,1043.94 205.5,1039.5C 210.672,1031.16 215.338,1022.49 219.5,1013.5C 238.656,980.53 257.323,947.196 275.5,913.5C 284.581,899.014 292.914,884.014 300.5,868.5C 299.933,866.018 299.266,863.684 298.5,861.5C 274.313,753.989 289.313,652.323 343.5,556.5C 347.461,550.306 351.628,544.306 356,538.5C 357.081,533.423 359.748,529.423 364,526.5C 365.839,522.827 368.006,519.494 370.5,516.5C 416.382,456.734 473.715,411.568 542.5,381C 557.672,374.288 573.339,368.955 589.5,365C 597.23,351.414 605.23,337.914 613.5,324.5C 617.425,319.2 621.258,313.867 625,308.5C 629.774,297.927 635.774,288.26 643,279.5C 644.407,276.137 645.907,272.803 647.5,269.5C 654.247,260.916 660.413,251.916 666,242.5C 666.311,240.444 666.811,238.444 667.5,236.5C 681.05,215.925 693.884,194.925 706,173.5C 707.494,172.253 708.994,172.253 710.5,173.5C 713.186,176.311 715.519,179.311 717.5,182.5 Z"/></g>
|
||||
+<g><path style="opacity:0.057" fill="#ecdf18" d="M 717.5,182.5 C 715.519,179.311 713.186,176.311 710.5,173.5C 708.994,172.253 707.494,172.253 706,173.5C 693.884,194.925 681.05,215.925 667.5,236.5C 666.811,238.444 666.311,240.444 666,242.5C 660.413,251.916 654.247,260.916 647.5,269.5C 645.907,272.803 644.407,276.137 643,279.5C 635.774,288.26 629.774,297.927 625,308.5C 621.258,313.867 617.425,319.2 613.5,324.5C 605.23,337.914 597.23,351.414 589.5,365C 573.339,368.955 557.672,374.288 542.5,381C 473.715,411.568 416.382,456.734 370.5,516.5C 370.629,514.784 370.962,513.117 371.5,511.5C 411.77,458.386 461.77,417.219 521.5,388C 543.036,378.155 565.036,369.488 587.5,362C 626.95,299.212 665.95,236.046 704.5,172.5C 707.167,171.167 709.833,171.167 712.5,172.5C 715.081,175.404 716.748,178.737 717.5,182.5 Z"/></g>
|
||||
+<g><path style="opacity:0.093" fill="#ece018" d="M 717.5,182.5 C 724.112,191.101 730.278,200.101 736,209.5C 766.572,260.563 797.738,311.23 829.5,361.5C 842.569,366.938 855.902,371.772 869.5,376C 886.41,383.122 902.744,391.289 918.5,400.5C 918.5,401.167 918.167,401.5 917.5,401.5C 915.518,401.005 913.518,400.505 911.5,400C 906.981,396.275 901.981,393.275 896.5,391C 894.048,390.71 891.715,390.044 889.5,389C 886.623,385.81 882.957,384.143 878.5,384C 878.002,382.842 877.169,382.009 876,381.5C 860.427,375.336 844.594,370.003 828.5,365.5C 821.742,353.852 814.742,342.185 807.5,330.5C 798.166,318.134 790.166,305.134 783.5,291.5C 777.722,285.018 773.055,277.685 769.5,269.5C 768.197,267.623 766.697,265.957 765,264.5C 764.625,262 763.792,259.667 762.5,257.5C 759.095,252.728 755.595,248.061 752,243.5C 751.399,240.363 750.399,237.363 749,234.5C 738.541,220.569 729.374,205.903 721.5,190.5C 718.106,188.891 716.773,186.224 717.5,182.5 Z"/></g>
|
||||
+<g><path style="opacity:1" fill="#fdfd00" d="M 708.5,295.5 C 710.451,296.716 711.951,298.382 713,300.5C 742.151,349.467 771.818,398.134 802,446.5C 904.901,479.736 977.568,546.069 1020,645.5C 1044.44,709.71 1049.1,775.376 1034,842.5C 1033.33,845.5 1033.33,848.5 1034,851.5C 1061.47,900.436 1088.97,949.436 1116.5,998.5C 1116.02,999.478 1115.36,1000.31 1114.5,1001C 1055.46,1003.3 996.464,1006.3 937.5,1010C 842.928,1091.3 734.928,1116.3 613.5,1085C 563.341,1070.08 519.007,1045.08 480.5,1010C 421.531,1006.38 362.531,1003.38 303.5,1001C 301.022,999.062 300.855,996.895 303,994.5C 330.253,946.328 357.253,897.995 384,849.5C 358.519,733.144 384.852,630.477 463,541.5C 505.106,496.852 555.606,465.519 614.5,447.5C 646.288,397.06 677.621,346.394 708.5,295.5 Z"/></g>
|
||||
+<g><path style="opacity:0.076" fill="#e2d51a" d="M 918.5,400.5 C 967.206,428.035 1009.04,463.701 1044,507.5C 1045.09,509.078 1045.93,510.745 1046.5,512.5C 1047.04,514.117 1047.37,515.784 1047.5,517.5C 1040.6,508.658 1033.43,499.992 1026,491.5C 1025.67,490.5 1025.33,489.5 1025,488.5C 1019.55,484.887 1015.38,480.22 1012.5,474.5C 1009.1,474.269 1007.1,472.436 1006.5,469C 1005.5,468.667 1004.5,468.333 1003.5,468C 991.922,456.583 979.922,445.75 967.5,435.5C 955.25,427.885 943.583,419.552 932.5,410.5C 928.054,408.114 923.387,405.947 918.5,404C 917.748,403.329 917.414,402.496 917.5,401.5C 918.167,401.5 918.5,401.167 918.5,400.5 Z"/></g>
|
||||
+<g><path style="opacity:1" fill="#30302f" d="M 694.5,567.5 C 771.354,565.01 831.521,595.343 875,658.5C 913.837,725.131 915.503,792.798 880,861.5C 836.437,931.265 773.271,964.098 690.5,960C 617.969,951.138 564.802,914.305 531,849.5C 502.746,785.583 505.746,723.25 540,662.5C 576.413,606.246 627.913,574.579 694.5,567.5 Z"/></g>
|
||||
+<g><path style="opacity:0.064" fill="#f6e716" d="M 1046.5,512.5 C 1055.58,523.094 1063.74,534.427 1071,546.5C 1129.23,644.638 1146.23,749.971 1122,862.5C 1121.5,863 1121,863.5 1120.5,864C 1129.42,878.329 1137.42,893.163 1144.5,908.5C 1143.18,908.67 1142.01,908.337 1141,907.5C 1140.49,905.122 1139.82,902.788 1139,900.5C 1136.15,896.81 1133.82,892.81 1132,888.5C 1130.99,889.337 1129.82,889.67 1128.5,889.5C 1125.11,884.395 1121.94,879.062 1119,873.5C 1118.83,871.833 1118.67,870.167 1118.5,868.5C 1147.5,739.989 1123.83,622.989 1047.5,517.5C 1047.37,515.784 1047.04,514.117 1046.5,512.5 Z"/></g>
|
||||
+<g><path style="opacity:0.239" fill="#dccf1d" d="M 298.5,861.5 C 299.266,863.684 299.933,866.018 300.5,868.5C 292.914,884.014 284.581,899.014 275.5,913.5C 274.782,911.863 274.616,910.197 275,908.5C 276.732,906.717 278.232,904.717 279.5,902.5C 279.291,901.914 278.957,901.414 278.5,901C 281.22,896.324 283.72,891.491 286,886.5C 286.777,888.701 287.944,888.701 289.5,886.5C 288.691,884.929 288.357,883.262 288.5,881.5C 289.552,881.649 290.552,881.483 291.5,881C 292.104,878.532 293.438,876.532 295.5,875C 294.703,872.426 295.537,870.259 298,868.5C 298.497,866.19 298.663,863.857 298.5,861.5 Z"/></g>
|
||||
+<g><path style="opacity:0.12" fill="#ecde19" d="M 1144.5,908.5 C 1167.91,948.307 1190.58,988.641 1212.5,1029.5C 1212.34,1032.55 1213.01,1035.38 1214.5,1038C 1214.17,1038.33 1213.83,1038.67 1213.5,1039C 1216.12,1039.58 1216.79,1040.91 1215.5,1043C 1217.86,1043.59 1218.86,1045.09 1218.5,1047.5C 1222.61,1048.49 1223.27,1050.49 1220.5,1053.5C 1219.57,1052.06 1218.4,1050.73 1217,1049.5C 1187.19,996.21 1157.69,942.876 1128.5,889.5C 1129.82,889.67 1130.99,889.337 1132,888.5C 1133.82,892.81 1136.15,896.81 1139,900.5C 1139.82,902.788 1140.49,905.122 1141,907.5C 1142.01,908.337 1143.18,908.67 1144.5,908.5 Z"/></g>
|
||||
+<g><path style="opacity:0.282" fill="#d5c91e" d="M 219.5,1013.5 C 215.338,1022.49 210.672,1031.16 205.5,1039.5C 205.756,1037.21 205.423,1035.04 204.5,1033C 209.614,1028.62 213.114,1023.12 215,1016.5C 216.271,1015.12 217.771,1014.12 219.5,1013.5 Z"/></g>
|
||||
+<g><path style="opacity:0.03" fill="#f3e517" d="M 371.5,511.5 C 370.962,513.117 370.629,514.784 370.5,516.5C 368.006,519.494 365.839,522.827 364,526.5C 359.748,529.423 357.081,533.423 356,538.5C 351.628,544.306 347.461,550.306 343.5,556.5C 289.313,652.323 274.313,753.989 298.5,861.5C 298.663,863.857 298.497,866.19 298,868.5C 295.537,870.259 294.703,872.426 295.5,875C 293.438,876.532 292.104,878.532 291.5,881C 290.552,881.483 289.552,881.649 288.5,881.5C 288.357,883.262 288.691,884.929 289.5,886.5C 287.944,888.701 286.777,888.701 286,886.5C 283.72,891.491 281.22,896.324 278.5,901C 278.957,901.414 279.291,901.914 279.5,902.5C 278.232,904.717 276.732,906.717 275,908.5C 274.616,910.197 274.782,911.863 275.5,913.5C 257.323,947.196 238.656,980.53 219.5,1013.5C 217.771,1014.12 216.271,1015.12 215,1016.5C 213.114,1023.12 209.614,1028.62 204.5,1033C 205.423,1035.04 205.756,1037.21 205.5,1039.5C 203.775,1043.94 201.275,1047.94 198,1051.5C 193.843,1063.01 197.677,1066.34 209.5,1061.5C 223.089,1063.38 236.756,1064.54 250.5,1065C 305.479,1068.22 360.479,1070.72 415.5,1072.5C 418.495,1073.66 421.495,1074.83 424.5,1076C 428.5,1080.06 432.5,1084.06 436.5,1088C 437.5,1088.33 438.5,1088.67 439.5,1089C 449.689,1098.8 461.022,1107.13 473.5,1114C 475.914,1117.45 479.247,1119.45 483.5,1120C 488.758,1122.98 493.758,1126.48 498.5,1130.5C 583.89,1177.51 674.89,1194.18 771.5,1180.5C 777.4,1180.12 783.067,1179.12 788.5,1177.5C 822.91,1171.64 855.91,1161.47 887.5,1147C 888.833,1146.33 889.833,1145.33 890.5,1144C 894.681,1143.91 898.015,1142.24 900.5,1139C 905.87,1135.98 911.537,1133.48 917.5,1131.5C 923.378,1126.31 930.045,1122.31 937.5,1119.5C 946.996,1111.65 956.996,1104.65 967.5,1098.5C 977.536,1088.28 988.536,1079.45 1000.5,1072C 1074,1069.95 1147.34,1066.45 1220.5,1061.5C 1221.61,1058.84 1221.61,1056.18 1220.5,1053.5C 1223.27,1050.49 1222.61,1048.49 1218.5,1047.5C 1218.86,1045.09 1217.86,1043.59 1215.5,1043C 1216.79,1040.91 1216.12,1039.58 1213.5,1039C 1213.83,1038.67 1214.17,1038.33 1214.5,1038C 1213.01,1035.38 1212.34,1032.55 1212.5,1029.5C 1218.28,1038.74 1223.78,1048.08 1229,1057.5C 1231.4,1062.49 1229.9,1065.99 1224.5,1068C 1221.44,1068.26 1218.44,1068.76 1215.5,1069.5C 1213.83,1068.83 1212.17,1068.17 1210.5,1067.5C 1180.44,1069.63 1150.44,1071.63 1120.5,1073.5C 1110.81,1073.47 1101.15,1073.47 1091.5,1073.5C 1059.05,1075.76 1026.71,1077.6 994.5,1079C 913.217,1150.34 818.05,1186.51 709,1187.5C 599.827,1186.84 504.661,1150.84 423.5,1079.5C 400.74,1077.32 377.74,1076.15 354.5,1076C 336.48,1074.62 318.647,1073.45 301,1072.5C 299.699,1072.96 298.366,1073.29 297,1073.5C 267.186,1071.67 237.353,1069.67 207.5,1067.5C 201.897,1069.4 196.231,1069.24 190.5,1067C 188.406,1065.74 187.406,1063.91 187.5,1061.5C 188.453,1057.6 189.953,1053.93 192,1050.5C 227.124,988.791 261.957,926.791 296.5,864.5C 271.3,753.389 287.133,648.723 344,550.5C 348.667,543.167 353.333,535.833 358,528.5C 362.418,522.603 366.918,516.936 371.5,511.5 Z"/></g>
|
||||
+<g><path style="opacity:1" fill="#4a4730" d="M 788.5,1177.5 C 783.067,1179.12 777.4,1180.12 771.5,1180.5C 772.952,1179.7 774.618,1179.2 776.5,1179C 774.5,1178.67 772.5,1178.33 770.5,1178C 776.486,1176.39 782.486,1176.22 788.5,1177.5 Z"/></g>
|
||||
</svg>
|
295
repos_extra/frontpage/diff_min.patch
Normal file
295
repos_extra/frontpage/diff_min.patch
Normal file
@@ -0,0 +1,295 @@
|
||||
diff --git a/packages/frontpage/app/(app)/layout.tsx b/packages/frontpage/app/(app)/layout.tsx
|
||||
index 232502d..8340a9f 100644
|
||||
--- a/packages/frontpage/app/(app)/layout.tsx
|
||||
+++ b/packages/frontpage/app/(app)/layout.tsx
|
||||
@@ -31,7 +31,8 @@ export default async function Layout({
|
||||
}) {
|
||||
const session = await getSession();
|
||||
return (
|
||||
- <div className="container mx-auto px-4 md:px-6 pt-4 pb-8 md:py-12 max-w-3xl">
|
||||
+ <div className="container mx-auto px-4 md:px-6 pt-4 pb-8 md:py-12">
|
||||
+ {/* <div className="container mx-auto px-4 md:px-6 pt-4 pb-8 md:py-12 max-w-3xl"> */}
|
||||
<div className="flex place-content-between items-center mb-8">
|
||||
<Link href="/">
|
||||
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||
@@ -52,6 +53,8 @@ export default async function Layout({
|
||||
|
||||
<main className="mb-6">{children}</main>
|
||||
|
||||
+ <footer> <span className="flex flex-row-reverse"><a href="https://github.com/likeandscribe/frontpage/" className="text-blue-600"><img src="https://raw.githubusercontent.com/likeandscribe/frontpage/refs/heads/main/packages/frontpage/public/frontpage-logo.svg" alt="Frontpage | MIT License | https://github.com/likeandscribe/frontpage/blob/main/LICENSE" className="h-10" /></a></span> <span className="flex justify-center items-center text-gray-500 dark:text-gray-400"> <a href={`https://bsky.app/profile/${FRONTPAGE_ATPROTO_HANDLE}`} className="font-medium text-indigo-600 hover:text-indigo-500 dark:text-indigo-400 dark:hover:text-indigo-300" > @syui.ai </a> </span> </footer>
|
||||
+ {/*
|
||||
<footer className="flex justify-between items-center text-gray-500 dark:text-gray-400">
|
||||
<p>
|
||||
Made by{" "}
|
||||
@@ -59,10 +62,11 @@ export default async function Layout({
|
||||
href={`https://bsky.app/profile/${FRONTPAGE_ATPROTO_HANDLE}`}
|
||||
className="font-medium text-indigo-600 hover:text-indigo-500 dark:text-indigo-400 dark:hover:text-indigo-300"
|
||||
>
|
||||
- @frontpage.fyi <OpenInNewWindowIcon className="inline" />
|
||||
+ @o.syui.ai <OpenInNewWindowIcon className="inline" />
|
||||
</a>
|
||||
</p>
|
||||
</footer>
|
||||
+ */}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -141,8 +145,8 @@ async function LoginOrLogout() {
|
||||
}
|
||||
|
||||
return (
|
||||
- <Button variant="outline" asChild>
|
||||
- <Link href="/login">Login</Link>
|
||||
+ <Button asChild>
|
||||
+ <Link href="/login">@</Link>
|
||||
</Button>
|
||||
);
|
||||
}
|
||||
diff --git a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx
|
||||
index 71dd664..46ab83a 100644
|
||||
--- a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx
|
||||
+++ b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/_lib/comment-client.tsx
|
||||
@@ -283,16 +283,16 @@ export function NewComment({
|
||||
ref={textAreaRef}
|
||||
placeholder="Write a comment..."
|
||||
disabled={isPending}
|
||||
- className="resize-y flex-1"
|
||||
+ className=""
|
||||
/>
|
||||
<div className="w-full flex justify-between">
|
||||
<InputLengthIndicator
|
||||
length={input.length}
|
||||
maxLength={MAX_COMMENT_LENGTH}
|
||||
/>
|
||||
- <div className="flex gap-2">
|
||||
+ <div className="w-full">
|
||||
{extraButton}
|
||||
- <Button type="submit" disabled={isPending}>
|
||||
+ <Button type="submit" disabled={isPending} className="w-full">
|
||||
{isPending ? <Spinner /> : "Post"}
|
||||
</Button>
|
||||
</div>
|
||||
diff --git a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx
|
||||
index 517c871..ef73ecb 100644
|
||||
--- a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx
|
||||
+++ b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/layout.tsx
|
||||
@@ -29,7 +29,7 @@ export default async function Post(props: {
|
||||
}
|
||||
|
||||
return (
|
||||
- <main className="mx-auto max-w-4xl space-y-6">
|
||||
+ <main className="mx-auto space-y-1">
|
||||
<PostCard
|
||||
author={post.authorDid}
|
||||
createdAt={post.createdAt}
|
||||
diff --git a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx
|
||||
index a83f810..11003f9 100644
|
||||
--- a/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx
|
||||
+++ b/packages/frontpage/app/(app)/post/[postAuthor]/[postRkey]/page.tsx
|
||||
@@ -20,7 +20,7 @@ export async function generateMetadata(props: {
|
||||
title: post.title,
|
||||
description: "Discuss this post on Frontpage.",
|
||||
alternates: {
|
||||
- canonical: `https://frontpage.fyi${path}`,
|
||||
+ canonical: `https://o.syui.ai${path}`,
|
||||
},
|
||||
openGraph: {
|
||||
title: post.title,
|
||||
@@ -28,7 +28,7 @@ export async function generateMetadata(props: {
|
||||
type: "article",
|
||||
publishedTime: post.createdAt.toISOString(),
|
||||
authors: [`@${handle}`],
|
||||
- url: `https://frontpage.fyi${path}`,
|
||||
+ url: `https://o.syui.ai${path}`,
|
||||
images: [
|
||||
{
|
||||
url: `${path}/og-image`,
|
||||
@@ -45,6 +45,8 @@ export default async function Post(props: { params: Promise<PostPageParams> }) {
|
||||
|
||||
return (
|
||||
<>
|
||||
+
|
||||
+ {/*
|
||||
<LinkAlternateAtUri
|
||||
authority={authorDid}
|
||||
collection={PostCollection}
|
||||
@@ -65,6 +67,71 @@ export default async function Post(props: { params: Promise<PostPageParams> }) {
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
+ */}
|
||||
+
|
||||
+ {params.postAuthor === "syui.ai" && (post.title.includes("live") || post.title.includes("video")) && (
|
||||
+ <div className="flex py-10 max-h-[720px] md:h-screen space-x-8">
|
||||
+ <iframe className="w-full h-[300px] sm:h-[450px] md:h-[630px]
|
||||
+ [&::-webkit-scrollbar]:w-2
|
||||
+ [&::-webkit-scrollbar-track]:bg-gray-100
|
||||
+ [&::-webkit-scrollbar-thumb]:bg-gray-300
|
||||
+ dark:[&::-webkit-scrollbar-track]:bg-neutral-700
|
||||
+ dark:[&::-webkit-scrollbar-thumb]:bg-neutral-500"
|
||||
+ title="stream" src={post.url}>
|
||||
+ </iframe>
|
||||
+ <div className="hidden w-64 hidden-scrollbar md:block md:overflow-y-scroll
|
||||
+ max-h-[600px]
|
||||
+ [&::-webkit-scrollbar]:w-2
|
||||
+ [&::-webkit-scrollbar-track]:bg-gray-100
|
||||
+ [&::-webkit-scrollbar-thumb]:bg-gray-300
|
||||
+ dark:[&::-webkit-scrollbar-track]:bg-neutral-700
|
||||
+ dark:[&::-webkit-scrollbar-thumb]:bg-neutral-500">
|
||||
+ <LinkAlternateAtUri
|
||||
+ authority={authorDid}
|
||||
+ collection={PostCollection}
|
||||
+ rkey={post.rkey}
|
||||
+ />
|
||||
+ <NewComment postRkey={post.rkey} postAuthorDid={authorDid} />
|
||||
+ {comments.map((comment) => (
|
||||
+ <Comment
|
||||
+ key={comment.id}
|
||||
+ comment={comment}
|
||||
+ level={0}
|
||||
+ postAuthorParam={params.postAuthor}
|
||||
+ postRkey={post.rkey}
|
||||
+ />
|
||||
+ ))}
|
||||
+ </div>
|
||||
+ </div>
|
||||
+ )}
|
||||
+ <LinkAlternateAtUri
|
||||
+ authority={authorDid}
|
||||
+ collection={PostCollection}
|
||||
+ rkey={post.rkey}
|
||||
+ />
|
||||
+ {post.status === "live" ? (
|
||||
+ <NewComment postRkey={post.rkey} postAuthorDid={authorDid} />
|
||||
+ ) : (
|
||||
+ <Alert>
|
||||
+ <AlertTitle>This post has been deleted</AlertTitle>
|
||||
+ <AlertDescription>
|
||||
+ Deleted posts cannot receive new comments.
|
||||
+ </AlertDescription>
|
||||
+ </Alert>
|
||||
+ )}
|
||||
+ <div className="flex flex-col gap-6 bg-white">
|
||||
+ {comments.map((comment) => (
|
||||
+ <Comment
|
||||
+ key={comment.id}
|
||||
+ comment={comment}
|
||||
+ level={0}
|
||||
+ postAuthorParam={params.postAuthor}
|
||||
+ postRkey={post.rkey}
|
||||
+ />
|
||||
+ ))}
|
||||
+
|
||||
+ </div>
|
||||
+
|
||||
</>
|
||||
);
|
||||
}
|
||||
diff --git a/packages/frontpage/app/(auth)/login/page.tsx b/packages/frontpage/app/(auth)/login/page.tsx
|
||||
index e3165f9..4d661ce 100644
|
||||
--- a/packages/frontpage/app/(auth)/login/page.tsx
|
||||
+++ b/packages/frontpage/app/(auth)/login/page.tsx
|
||||
@@ -14,17 +14,15 @@ export default async function LoginPage() {
|
||||
<div className="w-full max-w-md space-y-6">
|
||||
<div className="text-center">
|
||||
<h2 className="text-3xl font-bold tracking-tight text-gray-900 dark:text-gray-100">
|
||||
- Sign in to Frontpage
|
||||
+ Login with <span className="text-blue-600">at</span>proto
|
||||
</h2>
|
||||
<p className="mt-2 text-sm text-gray-600 dark:text-gray-400">
|
||||
- Don't have an account?{" "}
|
||||
<a
|
||||
href="https://bsky.app/"
|
||||
className="font-medium text-indigo-600 hover:text-indigo-500 dark:text-indigo-400 dark:hover:text-indigo-300"
|
||||
>
|
||||
- Sign up on Bluesky
|
||||
- </a>
|
||||
- , then return here to login.
|
||||
+ Bluesky
|
||||
+ </a> oauth login
|
||||
</p>
|
||||
</div>
|
||||
<LoginForm />
|
||||
diff --git a/packages/frontpage/app/globals.css b/packages/frontpage/app/globals.css
|
||||
index 9abc86b..464377c 100644
|
||||
--- a/packages/frontpage/app/globals.css
|
||||
+++ b/packages/frontpage/app/globals.css
|
||||
@@ -13,8 +13,12 @@
|
||||
--popover: 0 0% 100%;
|
||||
--popover-foreground: 222.2 84% 4.9%;
|
||||
|
||||
+ /*
|
||||
--primary: 222.2 47.4% 11.2%;
|
||||
- --primary-foreground: 210 40% 98%;
|
||||
+ --primary-foreground: 210 40% 98%; */
|
||||
+
|
||||
+ --primary: 210 100% 50%; /* HSLカラーで青色を定義 */
|
||||
+ --primary-foreground: 0 0% 100%; /* 白色のテキスト */
|
||||
|
||||
--secondary: 210 40% 96.1%;
|
||||
--secondary-foreground: 222.2 47.4% 11.2%;
|
||||
@@ -67,8 +71,12 @@
|
||||
--popover: 222.2 84% 4.9%;
|
||||
--popover-foreground: 210 40% 98%;
|
||||
|
||||
- --primary: 210 40% 98%;
|
||||
- --primary-foreground: 222.2 47.4% 11.2%;
|
||||
+ /*--primary: 210 40% 98%;
|
||||
+ *--primary-foreground: 222.2 47.4% 11.2%;*/
|
||||
+
|
||||
+ /* ダークモード用に少し明るい青 */
|
||||
+ --primary: 210 100% 60%;
|
||||
+ --primary-foreground: 0 0% 100%;
|
||||
|
||||
--secondary: 217.2 32.6% 17.5%;
|
||||
--secondary-foreground: 210 40% 98%;
|
||||
@@ -89,3 +97,7 @@
|
||||
--input: 217.2 32.6% 17.5%;
|
||||
--ring: 215.4 16.3% 46.9%;
|
||||
}
|
||||
diff --git a/packages/frontpage/app/layout.tsx b/packages/frontpage/app/layout.tsx
|
||||
index e8b5a39..a9fa945 100644
|
||||
--- a/packages/frontpage/app/layout.tsx
|
||||
+++ b/packages/frontpage/app/layout.tsx
|
||||
@@ -22,13 +22,13 @@ const sourceSans = Source_Sans_3({
|
||||
});
|
||||
|
||||
export const metadata: Metadata = {
|
||||
- title: "Frontpage",
|
||||
+ title: "o.syui.ai",
|
||||
description: "A federated link aggregator. Your frontpage to the internet.",
|
||||
openGraph: {
|
||||
title: "Frontpage",
|
||||
description: "Your frontpage to the internet.",
|
||||
type: "website",
|
||||
- siteName: "frontpage.fyi",
|
||||
+ siteName: "o.syui.ai",
|
||||
},
|
||||
};
|
||||
|
||||
@@ -53,11 +53,7 @@ export default function Layout({ children }: { children: React.ReactNode }) {
|
||||
<SpeedInsights />
|
||||
<Analytics />
|
||||
</ThemeProvider>
|
||||
- <script
|
||||
- defer
|
||||
- src="https://static.cloudflareinsights.com/beacon.min.js"
|
||||
- data-cf-beacon='{"token": "5e64b32c70b34b1583d863032b03d9ad"}'
|
||||
- />
|
||||
+ {/* <script defer src="https://static.cloudflareinsights.com/beacon.min.js" data-cf-beacon='{"token": "5e64b32c70b34b1583d863032b03d9ad"}' /> */}
|
||||
</body>
|
||||
</html>
|
||||
);
|
||||
diff --git a/packages/frontpage/app/(app)/_components/post-card.tsx b/packages/frontpage/app/(app)/_components/post-card.tsx
|
||||
index c0720e5..e619a5f 100644
|
||||
--- a/packages/frontpage/app/(app)/_components/post-card.tsx
|
||||
+++ b/packages/frontpage/app/(app)/_components/post-card.tsx
|
||||
@@ -83,7 +83,7 @@ export async function PostCard({
|
||||
votes={votes}
|
||||
/>
|
||||
</div>
|
||||
- <div className="w-full">
|
||||
+ <div className="w-full overflow-hidden">
|
||||
<h2 className="mb-1 text-xl">
|
||||
<a
|
||||
href={url}
|
30
repos_extra/frontpage/lexicons/ai/syui/o/comment.json
Normal file
30
repos_extra/frontpage/lexicons/ai/syui/o/comment.json
Normal file
@@ -0,0 +1,30 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "ai.syui.o.comment",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "record",
|
||||
"description": "Record containing a Frontpage comment.",
|
||||
"key": "tid",
|
||||
"record": {
|
||||
"type": "object",
|
||||
"required": ["content", "createdAt", "post"],
|
||||
"properties": {
|
||||
"content": {
|
||||
"type": "string",
|
||||
"maxLength": 100000,
|
||||
"maxGraphemes": 10000,
|
||||
"description": "The content of the comment."
|
||||
},
|
||||
"createdAt": {
|
||||
"type": "string",
|
||||
"format": "datetime",
|
||||
"description": "Client-declared timestamp when this comment was originally created."
|
||||
},
|
||||
"parent": { "type": "ref", "ref": "com.atproto.repo.strongRef" },
|
||||
"post": { "type": "ref", "ref": "com.atproto.repo.strongRef" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
33
repos_extra/frontpage/lexicons/ai/syui/o/post.json
Normal file
33
repos_extra/frontpage/lexicons/ai/syui/o/post.json
Normal file
@@ -0,0 +1,33 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "ai.syui.o.post",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "record",
|
||||
"description": "Record containing a Frontpage post.",
|
||||
"key": "tid",
|
||||
"record": {
|
||||
"type": "object",
|
||||
"required": ["title", "url", "createdAt"],
|
||||
"properties": {
|
||||
"title": {
|
||||
"type": "string",
|
||||
"maxLength": 3000,
|
||||
"maxGraphemes": 300,
|
||||
"description": "The title of the post."
|
||||
},
|
||||
"url": {
|
||||
"type": "string",
|
||||
"format": "uri",
|
||||
"description": "The URL of the post."
|
||||
},
|
||||
"createdAt": {
|
||||
"type": "string",
|
||||
"format": "datetime",
|
||||
"description": "Client-declared timestamp when this post was originally created."
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
23
repos_extra/frontpage/lexicons/ai/syui/o/vote.json
Normal file
23
repos_extra/frontpage/lexicons/ai/syui/o/vote.json
Normal file
@@ -0,0 +1,23 @@
|
||||
{
|
||||
"lexicon": 1,
|
||||
"id": "ai.syui.o.vote",
|
||||
"defs": {
|
||||
"main": {
|
||||
"type": "record",
|
||||
"description": "Record containing a Frontpage vote.",
|
||||
"key": "tid",
|
||||
"record": {
|
||||
"type": "object",
|
||||
"required": ["subject", "createdAt"],
|
||||
"properties": {
|
||||
"subject": { "type": "ref", "ref": "com.atproto.repo.strongRef" },
|
||||
"createdAt": {
|
||||
"type": "string",
|
||||
"format": "datetime",
|
||||
"description": "Client-declared timestamp when this vote was originally created."
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
5
repos_extra/frontpage/packages-rs/drainpipe/.env.local
Normal file
5
repos_extra/frontpage/packages-rs/drainpipe/.env.local
Normal file
@@ -0,0 +1,5 @@
|
||||
DATABASE_URL="drainpipe.db"
|
||||
FRONTPAGE_CONSUMER_URL="http://${cloudflared}/api/receive_hook"
|
||||
FRONTPAGE_CONSUMER_SECRET=`openssl ecparam --name secp256k1 --genkey --noout --outform DER | tail --bytes=+8 | head --bytes=32 | xxd --plain --cols 32`
|
||||
|
||||
#RELAY_URL=wss://syu.is
|
0
repos_extra/frontpage/packages-rs/drainpipe/.keep
Normal file
0
repos_extra/frontpage/packages-rs/drainpipe/.keep
Normal file
20
repos_extra/frontpage/packages-rs/drainpipe/Dockerfile
Normal file
20
repos_extra/frontpage/packages-rs/drainpipe/Dockerfile
Normal file
@@ -0,0 +1,20 @@
|
||||
FROM rust:1.78-alpine AS builder
|
||||
|
||||
RUN apk add libressl-dev musl-dev sqlite-dev
|
||||
|
||||
WORKDIR /usr/src/unravel
|
||||
COPY . .
|
||||
# TODO: Use cargo-chef to cache dependencies compilation independently of the binary
|
||||
RUN --mount=type=cache,target=/usr/local/cargo/registry \
|
||||
--mount=type=cache,target=/root/target \
|
||||
cargo build --release --package drainpipe && \
|
||||
# Move the release binary to a folder to be copied to the final image. It can't be copied directly from the target folder because it's in a cache mount
|
||||
# See https://gist.github.com/noelbundick/6922d26667616e2ba5c3aff59f0824cd?permalink_comment_id=4379948#gistcomment-4379948
|
||||
mv ./target/release /root
|
||||
|
||||
FROM alpine:3.14
|
||||
COPY --from=builder /root/release/drainpipe /
|
||||
|
||||
ENV DATABASE_URL="/drainpipedata/drainpipe.db"
|
||||
|
||||
ENTRYPOINT ["/drainpipe"]
|
@@ -0,0 +1,9 @@
|
||||
services:
|
||||
drainpipe:
|
||||
build:
|
||||
dockerfile: ./packages-rs/drainpipe/Dockerfile
|
||||
context: ../../
|
||||
env_file:
|
||||
- ./.env.local
|
||||
volumes:
|
||||
- ./drainpipedata:/drainpipedata
|
10
repos_extra/frontpage/packages/atproto-browser/Dockerfile
Normal file
10
repos_extra/frontpage/packages/atproto-browser/Dockerfile
Normal file
@@ -0,0 +1,10 @@
|
||||
FROM node:20
|
||||
|
||||
RUN npm install -g pnpm
|
||||
WORKDIR /app
|
||||
RUN git clone https://github.com/likeandscribe/frontpage
|
||||
WORKDIR /app/frontpage/packages/atproto-browser
|
||||
RUN pnpm i
|
||||
RUN pnpm build
|
||||
|
||||
CMD [ "pnpm", "start"]
|
@@ -0,0 +1,8 @@
|
||||
services:
|
||||
atbrowser:
|
||||
build:
|
||||
context: .
|
||||
ports:
|
||||
- "3000:3000"
|
||||
environment:
|
||||
- WATCHPACK_POLLING=true
|
14
repos_extra/frontpage/packages/frontpage/.env.local
Normal file
14
repos_extra/frontpage/packages/frontpage/.env.local
Normal file
@@ -0,0 +1,14 @@
|
||||
PRIVATE_JWK=`pnpm exec tsx ./scripts/generate-jwk.mts`
|
||||
PUBLIC_JWK=`pnpm exec tsx ./scripts/generate-jwk.mts`
|
||||
|
||||
TURSO_CONNECTION_URL=libsql://xxx.turso.io
|
||||
#TURSO_CONNECTION_URL=`turso db shell xxx-xxx`
|
||||
TURSO_AUTH_TOKEN=`turso db tokens create xxx-xxx`
|
||||
|
||||
DRAINPIPE_CONSUMER_SECRET=`openssl ecparam --name secp256k1 --genkey --noout --outform DER | tail --bytes=+8 | head --bytes=32 | xxd --plain --cols 32`
|
||||
VERCEL_PROJECT_PRODUCTION_URL=example.com
|
||||
VERCEL_BRANCH_URL=example.com
|
||||
|
||||
#DRAINPIPE_CONSUMER_SECRET=secret
|
||||
#TURSO_CONNECTION_URL=libsql://turso.dev.unravel.fyi
|
||||
#PLC_DIRECTORY_URL=https://plc.dev.unravel.fyi
|
17
repos_extra/frontpage/packages/frontpage/Dockerfile
Normal file
17
repos_extra/frontpage/packages/frontpage/Dockerfile
Normal file
@@ -0,0 +1,17 @@
|
||||
FROM node:20
|
||||
|
||||
RUN npm install -g pnpm
|
||||
WORKDIR /app
|
||||
RUN git clone https://github.com/likeandscribe/frontpage
|
||||
WORKDIR /app/frontpage
|
||||
RUN pnpm i
|
||||
RUN pnpm exec turbo run --affected type-check
|
||||
|
||||
WORKDIR /app/frontpage/packages/frontpage
|
||||
COPY ./.env.local ./.env.local
|
||||
COPY ./app ./app
|
||||
COPY ./lib ./lib
|
||||
RUN pnpm run db:generate
|
||||
RUN pnpm run db:migrate
|
||||
RUN pnpm run build
|
||||
CMD [ "pnpm", "run", "start"]
|
@@ -0,0 +1,4 @@
|
||||
<Button asChild>
|
||||
<Link href="/post/game">Game</Link>
|
||||
</Button>
|
||||
|
@@ -0,0 +1,26 @@
|
||||
"use server";
|
||||
|
||||
import { DID } from "@/lib/data/atproto/did";
|
||||
import { getVerifiedHandle } from "@/lib/data/atproto/identity";
|
||||
import { putPost } from "@/lib/data/atproto/game";
|
||||
import { uncached_doesPostExist } from "@/lib/data/db/post";
|
||||
import { DataLayerError } from "@/lib/data/error";
|
||||
import { ensureUser } from "@/lib/data/user";
|
||||
import { redirect } from "next/navigation";
|
||||
|
||||
export async function newPostAction(_prevState: unknown) {
|
||||
"use server";
|
||||
const user = await ensureUser();
|
||||
const [handle] = await Promise.all([
|
||||
getVerifiedHandle(user.did),
|
||||
]);
|
||||
|
||||
const account = `at://did:plc:4hqjfn7m6n5hno3doamuhgef/ai.syui.game.user/${handle}`;
|
||||
try {
|
||||
const { rkey } = await putPost({ account });
|
||||
redirect(`https://at.syu.is/at/${user.did}/ai.syui.game/self`);
|
||||
} catch (error) {
|
||||
if (!(error instanceof DataLayerError)) throw error;
|
||||
return { error: "Failed to create post" };
|
||||
}
|
||||
}
|
@@ -0,0 +1,36 @@
|
||||
"use client";
|
||||
|
||||
import { startTransition, useActionState, useId, useState } from "react";
|
||||
import { newPostAction } from "./_action";
|
||||
import { Label } from "@/lib/components/ui/label";
|
||||
import { Input } from "@/lib/components/ui/input";
|
||||
import { Button } from "@/lib/components/ui/button";
|
||||
import { Alert, AlertDescription, AlertTitle } from "@/lib/components/ui/alert";
|
||||
import { Spinner } from "@/lib/components/ui/spinner";
|
||||
import { InputLengthIndicator } from "@/lib/components/input-length-indicator";
|
||||
|
||||
export function NewPostButton() {
|
||||
const [state, action, isPending] = useActionState(newPostAction, null);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
|
||||
const handleClick = async () => {
|
||||
try {
|
||||
const result = await action({});
|
||||
if (result && 'error' in result) {
|
||||
setError(result.error);
|
||||
}
|
||||
} catch (e) {
|
||||
console.error('Error creating post:', e);
|
||||
setError('予期しないエラーが発生しました');
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div>
|
||||
<button onClick={handleClick} disabled={isPending}>
|
||||
{isPending ? '投稿中...' : '新規投稿'}
|
||||
</button>
|
||||
{error && <p style={{ color: 'red' }}>{error}</p>}
|
||||
</div>
|
||||
);
|
||||
}
|
@@ -0,0 +1,17 @@
|
||||
import { Metadata } from "next";
|
||||
import { NewPostButton } from "./_client";
|
||||
|
||||
export const metadata: Metadata = {
|
||||
title: "New post | Frontpage",
|
||||
robots: "noindex, nofollow",
|
||||
};
|
||||
|
||||
export default function NewPost() {
|
||||
return (
|
||||
<main className="flex flex-col gap-3">
|
||||
<h2 className="text-3xl font-bold tracking-tight text-gray-900 dark:text-gray-100">
|
||||
</h2>
|
||||
<NewPostButton />
|
||||
</main>
|
||||
);
|
||||
}
|
8
repos_extra/frontpage/packages/frontpage/compose.yml
Normal file
8
repos_extra/frontpage/packages/frontpage/compose.yml
Normal file
@@ -0,0 +1,8 @@
|
||||
services:
|
||||
frontpage:
|
||||
build:
|
||||
context: .
|
||||
ports:
|
||||
- "3000:3000"
|
||||
environment:
|
||||
- WATCHPACK_POLLING=true
|
@@ -0,0 +1,62 @@
|
||||
import "server-only";
|
||||
import {
|
||||
atprotoPutRecord,
|
||||
atprotoCreateRecord,
|
||||
atprotoDeleteRecord,
|
||||
atprotoGetRecord,
|
||||
} from "./record";
|
||||
import { z } from "zod";
|
||||
import { DataLayerError } from "../error";
|
||||
import { DID, getPdsUrl } from "./did";
|
||||
|
||||
export const PostCollection = "ai.syui.game";
|
||||
|
||||
export const PostRecord = z.object({
|
||||
account: z.string(),
|
||||
createdAt: z.string(),
|
||||
});
|
||||
|
||||
export type Post = z.infer<typeof PostRecord>;
|
||||
|
||||
type PostInput = {
|
||||
account: string;
|
||||
};
|
||||
|
||||
export async function putPost({ account }: PostInput) {
|
||||
const record = { account, createdAt: new Date().toISOString() };
|
||||
PostRecord.parse(record);
|
||||
|
||||
const result = await atprotoPutRecord({
|
||||
rkey: "self",
|
||||
record,
|
||||
collection: PostCollection,
|
||||
});
|
||||
|
||||
return {
|
||||
rkey: "self",
|
||||
};
|
||||
}
|
||||
|
||||
export async function deletePost(rkey: string) {
|
||||
await atprotoDeleteRecord({
|
||||
rkey,
|
||||
collection: PostCollection,
|
||||
});
|
||||
}
|
||||
|
||||
export async function getPost({ rkey, repo }: { rkey: string; repo: DID }) {
|
||||
const service = await getPdsUrl(repo);
|
||||
|
||||
if (!service) {
|
||||
throw new DataLayerError("Failed to get service url");
|
||||
}
|
||||
|
||||
const { value } = await atprotoGetRecord({
|
||||
serviceEndpoint: service,
|
||||
repo,
|
||||
collection: PostCollection,
|
||||
rkey,
|
||||
});
|
||||
|
||||
return PostRecord.parse(value);
|
||||
}
|
@@ -0,0 +1,50 @@
|
||||
import "server-only";
|
||||
import { z } from "zod";
|
||||
import { ensureUser } from "../user";
|
||||
import { DataLayerError } from "../error";
|
||||
import { fetchAuthenticatedAtproto } from "@/lib/auth";
|
||||
import { AtUri } from "./uri";
|
||||
|
||||
const PutRecordResponse = z.object({
|
||||
uri: AtUri,
|
||||
cid: z.string(),
|
||||
});
|
||||
|
||||
type PutRecordInput = {
|
||||
rkey?: string;
|
||||
record: unknown;
|
||||
collection: string;
|
||||
};
|
||||
|
||||
export async function atprotoPutRecord({
|
||||
rkey,
|
||||
record,
|
||||
collection,
|
||||
}: PutRecordInput) {
|
||||
const user = await ensureUser();
|
||||
const pdsUrl = new URL(user.pdsUrl);
|
||||
pdsUrl.pathname = "/xrpc/com.atproto.repo.putRecord";
|
||||
|
||||
const response = await fetchAuthenticatedAtproto(pdsUrl.toString(), {
|
||||
method: "POST",
|
||||
headers: {
|
||||
"Content-Type": "application/json",
|
||||
},
|
||||
body: JSON.stringify({
|
||||
rkey: rkey || "self",
|
||||
repo: user.did,
|
||||
collection,
|
||||
validate: false,
|
||||
record: record,
|
||||
}),
|
||||
});
|
||||
|
||||
if (!response.ok) {
|
||||
throw new DataLayerError(`Failed to create record ${response.status}`, {
|
||||
cause: response,
|
||||
});
|
||||
}
|
||||
|
||||
return PutRecordResponse.parse(await response.json());
|
||||
}
|
||||
|
@@ -0,0 +1,16 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" version="1.1" width="1417px" height="1417px" style="shape-rendering:geometricPrecision; text-rendering:geometricPrecision; image-rendering:optimizeQuality; fill-rule:evenodd; clip-rule:evenodd" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||
<g><path style="opacity:1" fill="#303030" d="M 717.5,182.5 C 716.773,186.224 718.106,188.891 721.5,190.5C 729.374,205.903 738.541,220.569 749,234.5C 750.399,237.363 751.399,240.363 752,243.5C 755.595,248.061 759.095,252.728 762.5,257.5C 763.792,259.667 764.625,262 765,264.5C 766.697,265.957 768.197,267.623 769.5,269.5C 773.055,277.685 777.722,285.018 783.5,291.5C 790.166,305.134 798.166,318.134 807.5,330.5C 814.742,342.185 821.742,353.852 828.5,365.5C 844.594,370.003 860.427,375.336 876,381.5C 877.169,382.009 878.002,382.842 878.5,384C 882.957,384.143 886.623,385.81 889.5,389C 891.715,390.044 894.048,390.71 896.5,391C 901.981,393.275 906.981,396.275 911.5,400C 913.518,400.505 915.518,401.005 917.5,401.5C 917.414,402.496 917.748,403.329 918.5,404C 923.387,405.947 928.054,408.114 932.5,410.5C 943.583,419.552 955.25,427.885 967.5,435.5C 979.922,445.75 991.922,456.583 1003.5,468C 1004.5,468.333 1005.5,468.667 1006.5,469C 1007.1,472.436 1009.1,474.269 1012.5,474.5C 1015.38,480.22 1019.55,484.887 1025,488.5C 1025.33,489.5 1025.67,490.5 1026,491.5C 1033.43,499.992 1040.6,508.658 1047.5,517.5C 1123.83,622.989 1147.5,739.989 1118.5,868.5C 1118.67,870.167 1118.83,871.833 1119,873.5C 1121.94,879.062 1125.11,884.395 1128.5,889.5C 1157.69,942.876 1187.19,996.21 1217,1049.5C 1218.4,1050.73 1219.57,1052.06 1220.5,1053.5C 1221.61,1056.18 1221.61,1058.84 1220.5,1061.5C 1147.34,1066.45 1074,1069.95 1000.5,1072C 988.536,1079.45 977.536,1088.28 967.5,1098.5C 956.996,1104.65 946.996,1111.65 937.5,1119.5C 930.045,1122.31 923.378,1126.31 917.5,1131.5C 911.537,1133.48 905.87,1135.98 900.5,1139C 898.015,1142.24 894.681,1143.91 890.5,1144C 889.833,1145.33 888.833,1146.33 887.5,1147C 855.91,1161.47 822.91,1171.64 788.5,1177.5C 782.486,1176.22 776.486,1176.39 770.5,1178C 772.5,1178.33 774.5,1178.67 776.5,1179C 774.618,1179.2 772.952,1179.7 771.5,1180.5C 674.89,1194.18 583.89,1177.51 498.5,1130.5C 493.758,1126.48 488.758,1122.98 483.5,1120C 479.247,1119.45 475.914,1117.45 473.5,1114C 461.022,1107.13 449.689,1098.8 439.5,1089C 438.5,1088.67 437.5,1088.33 436.5,1088C 432.5,1084.06 428.5,1080.06 424.5,1076C 421.495,1074.83 418.495,1073.66 415.5,1072.5C 360.479,1070.72 305.479,1068.22 250.5,1065C 236.756,1064.54 223.089,1063.38 209.5,1061.5C 197.677,1066.34 193.843,1063.01 198,1051.5C 201.275,1047.94 203.775,1043.94 205.5,1039.5C 210.672,1031.16 215.338,1022.49 219.5,1013.5C 238.656,980.53 257.323,947.196 275.5,913.5C 284.581,899.014 292.914,884.014 300.5,868.5C 299.933,866.018 299.266,863.684 298.5,861.5C 274.313,753.989 289.313,652.323 343.5,556.5C 347.461,550.306 351.628,544.306 356,538.5C 357.081,533.423 359.748,529.423 364,526.5C 365.839,522.827 368.006,519.494 370.5,516.5C 416.382,456.734 473.715,411.568 542.5,381C 557.672,374.288 573.339,368.955 589.5,365C 597.23,351.414 605.23,337.914 613.5,324.5C 617.425,319.2 621.258,313.867 625,308.5C 629.774,297.927 635.774,288.26 643,279.5C 644.407,276.137 645.907,272.803 647.5,269.5C 654.247,260.916 660.413,251.916 666,242.5C 666.311,240.444 666.811,238.444 667.5,236.5C 681.05,215.925 693.884,194.925 706,173.5C 707.494,172.253 708.994,172.253 710.5,173.5C 713.186,176.311 715.519,179.311 717.5,182.5 Z"/></g>
|
||||
<g><path style="opacity:0.057" fill="#ecdf18" d="M 717.5,182.5 C 715.519,179.311 713.186,176.311 710.5,173.5C 708.994,172.253 707.494,172.253 706,173.5C 693.884,194.925 681.05,215.925 667.5,236.5C 666.811,238.444 666.311,240.444 666,242.5C 660.413,251.916 654.247,260.916 647.5,269.5C 645.907,272.803 644.407,276.137 643,279.5C 635.774,288.26 629.774,297.927 625,308.5C 621.258,313.867 617.425,319.2 613.5,324.5C 605.23,337.914 597.23,351.414 589.5,365C 573.339,368.955 557.672,374.288 542.5,381C 473.715,411.568 416.382,456.734 370.5,516.5C 370.629,514.784 370.962,513.117 371.5,511.5C 411.77,458.386 461.77,417.219 521.5,388C 543.036,378.155 565.036,369.488 587.5,362C 626.95,299.212 665.95,236.046 704.5,172.5C 707.167,171.167 709.833,171.167 712.5,172.5C 715.081,175.404 716.748,178.737 717.5,182.5 Z"/></g>
|
||||
<g><path style="opacity:0.093" fill="#ece018" d="M 717.5,182.5 C 724.112,191.101 730.278,200.101 736,209.5C 766.572,260.563 797.738,311.23 829.5,361.5C 842.569,366.938 855.902,371.772 869.5,376C 886.41,383.122 902.744,391.289 918.5,400.5C 918.5,401.167 918.167,401.5 917.5,401.5C 915.518,401.005 913.518,400.505 911.5,400C 906.981,396.275 901.981,393.275 896.5,391C 894.048,390.71 891.715,390.044 889.5,389C 886.623,385.81 882.957,384.143 878.5,384C 878.002,382.842 877.169,382.009 876,381.5C 860.427,375.336 844.594,370.003 828.5,365.5C 821.742,353.852 814.742,342.185 807.5,330.5C 798.166,318.134 790.166,305.134 783.5,291.5C 777.722,285.018 773.055,277.685 769.5,269.5C 768.197,267.623 766.697,265.957 765,264.5C 764.625,262 763.792,259.667 762.5,257.5C 759.095,252.728 755.595,248.061 752,243.5C 751.399,240.363 750.399,237.363 749,234.5C 738.541,220.569 729.374,205.903 721.5,190.5C 718.106,188.891 716.773,186.224 717.5,182.5 Z"/></g>
|
||||
<g><path style="opacity:1" fill="#fdfd00" d="M 708.5,295.5 C 710.451,296.716 711.951,298.382 713,300.5C 742.151,349.467 771.818,398.134 802,446.5C 904.901,479.736 977.568,546.069 1020,645.5C 1044.44,709.71 1049.1,775.376 1034,842.5C 1033.33,845.5 1033.33,848.5 1034,851.5C 1061.47,900.436 1088.97,949.436 1116.5,998.5C 1116.02,999.478 1115.36,1000.31 1114.5,1001C 1055.46,1003.3 996.464,1006.3 937.5,1010C 842.928,1091.3 734.928,1116.3 613.5,1085C 563.341,1070.08 519.007,1045.08 480.5,1010C 421.531,1006.38 362.531,1003.38 303.5,1001C 301.022,999.062 300.855,996.895 303,994.5C 330.253,946.328 357.253,897.995 384,849.5C 358.519,733.144 384.852,630.477 463,541.5C 505.106,496.852 555.606,465.519 614.5,447.5C 646.288,397.06 677.621,346.394 708.5,295.5 Z"/></g>
|
||||
<g><path style="opacity:0.076" fill="#e2d51a" d="M 918.5,400.5 C 967.206,428.035 1009.04,463.701 1044,507.5C 1045.09,509.078 1045.93,510.745 1046.5,512.5C 1047.04,514.117 1047.37,515.784 1047.5,517.5C 1040.6,508.658 1033.43,499.992 1026,491.5C 1025.67,490.5 1025.33,489.5 1025,488.5C 1019.55,484.887 1015.38,480.22 1012.5,474.5C 1009.1,474.269 1007.1,472.436 1006.5,469C 1005.5,468.667 1004.5,468.333 1003.5,468C 991.922,456.583 979.922,445.75 967.5,435.5C 955.25,427.885 943.583,419.552 932.5,410.5C 928.054,408.114 923.387,405.947 918.5,404C 917.748,403.329 917.414,402.496 917.5,401.5C 918.167,401.5 918.5,401.167 918.5,400.5 Z"/></g>
|
||||
<g><path style="opacity:1" fill="#30302f" d="M 694.5,567.5 C 771.354,565.01 831.521,595.343 875,658.5C 913.837,725.131 915.503,792.798 880,861.5C 836.437,931.265 773.271,964.098 690.5,960C 617.969,951.138 564.802,914.305 531,849.5C 502.746,785.583 505.746,723.25 540,662.5C 576.413,606.246 627.913,574.579 694.5,567.5 Z"/></g>
|
||||
<g><path style="opacity:0.064" fill="#f6e716" d="M 1046.5,512.5 C 1055.58,523.094 1063.74,534.427 1071,546.5C 1129.23,644.638 1146.23,749.971 1122,862.5C 1121.5,863 1121,863.5 1120.5,864C 1129.42,878.329 1137.42,893.163 1144.5,908.5C 1143.18,908.67 1142.01,908.337 1141,907.5C 1140.49,905.122 1139.82,902.788 1139,900.5C 1136.15,896.81 1133.82,892.81 1132,888.5C 1130.99,889.337 1129.82,889.67 1128.5,889.5C 1125.11,884.395 1121.94,879.062 1119,873.5C 1118.83,871.833 1118.67,870.167 1118.5,868.5C 1147.5,739.989 1123.83,622.989 1047.5,517.5C 1047.37,515.784 1047.04,514.117 1046.5,512.5 Z"/></g>
|
||||
<g><path style="opacity:0.239" fill="#dccf1d" d="M 298.5,861.5 C 299.266,863.684 299.933,866.018 300.5,868.5C 292.914,884.014 284.581,899.014 275.5,913.5C 274.782,911.863 274.616,910.197 275,908.5C 276.732,906.717 278.232,904.717 279.5,902.5C 279.291,901.914 278.957,901.414 278.5,901C 281.22,896.324 283.72,891.491 286,886.5C 286.777,888.701 287.944,888.701 289.5,886.5C 288.691,884.929 288.357,883.262 288.5,881.5C 289.552,881.649 290.552,881.483 291.5,881C 292.104,878.532 293.438,876.532 295.5,875C 294.703,872.426 295.537,870.259 298,868.5C 298.497,866.19 298.663,863.857 298.5,861.5 Z"/></g>
|
||||
<g><path style="opacity:0.12" fill="#ecde19" d="M 1144.5,908.5 C 1167.91,948.307 1190.58,988.641 1212.5,1029.5C 1212.34,1032.55 1213.01,1035.38 1214.5,1038C 1214.17,1038.33 1213.83,1038.67 1213.5,1039C 1216.12,1039.58 1216.79,1040.91 1215.5,1043C 1217.86,1043.59 1218.86,1045.09 1218.5,1047.5C 1222.61,1048.49 1223.27,1050.49 1220.5,1053.5C 1219.57,1052.06 1218.4,1050.73 1217,1049.5C 1187.19,996.21 1157.69,942.876 1128.5,889.5C 1129.82,889.67 1130.99,889.337 1132,888.5C 1133.82,892.81 1136.15,896.81 1139,900.5C 1139.82,902.788 1140.49,905.122 1141,907.5C 1142.01,908.337 1143.18,908.67 1144.5,908.5 Z"/></g>
|
||||
<g><path style="opacity:0.282" fill="#d5c91e" d="M 219.5,1013.5 C 215.338,1022.49 210.672,1031.16 205.5,1039.5C 205.756,1037.21 205.423,1035.04 204.5,1033C 209.614,1028.62 213.114,1023.12 215,1016.5C 216.271,1015.12 217.771,1014.12 219.5,1013.5 Z"/></g>
|
||||
<g><path style="opacity:0.03" fill="#f3e517" d="M 371.5,511.5 C 370.962,513.117 370.629,514.784 370.5,516.5C 368.006,519.494 365.839,522.827 364,526.5C 359.748,529.423 357.081,533.423 356,538.5C 351.628,544.306 347.461,550.306 343.5,556.5C 289.313,652.323 274.313,753.989 298.5,861.5C 298.663,863.857 298.497,866.19 298,868.5C 295.537,870.259 294.703,872.426 295.5,875C 293.438,876.532 292.104,878.532 291.5,881C 290.552,881.483 289.552,881.649 288.5,881.5C 288.357,883.262 288.691,884.929 289.5,886.5C 287.944,888.701 286.777,888.701 286,886.5C 283.72,891.491 281.22,896.324 278.5,901C 278.957,901.414 279.291,901.914 279.5,902.5C 278.232,904.717 276.732,906.717 275,908.5C 274.616,910.197 274.782,911.863 275.5,913.5C 257.323,947.196 238.656,980.53 219.5,1013.5C 217.771,1014.12 216.271,1015.12 215,1016.5C 213.114,1023.12 209.614,1028.62 204.5,1033C 205.423,1035.04 205.756,1037.21 205.5,1039.5C 203.775,1043.94 201.275,1047.94 198,1051.5C 193.843,1063.01 197.677,1066.34 209.5,1061.5C 223.089,1063.38 236.756,1064.54 250.5,1065C 305.479,1068.22 360.479,1070.72 415.5,1072.5C 418.495,1073.66 421.495,1074.83 424.5,1076C 428.5,1080.06 432.5,1084.06 436.5,1088C 437.5,1088.33 438.5,1088.67 439.5,1089C 449.689,1098.8 461.022,1107.13 473.5,1114C 475.914,1117.45 479.247,1119.45 483.5,1120C 488.758,1122.98 493.758,1126.48 498.5,1130.5C 583.89,1177.51 674.89,1194.18 771.5,1180.5C 777.4,1180.12 783.067,1179.12 788.5,1177.5C 822.91,1171.64 855.91,1161.47 887.5,1147C 888.833,1146.33 889.833,1145.33 890.5,1144C 894.681,1143.91 898.015,1142.24 900.5,1139C 905.87,1135.98 911.537,1133.48 917.5,1131.5C 923.378,1126.31 930.045,1122.31 937.5,1119.5C 946.996,1111.65 956.996,1104.65 967.5,1098.5C 977.536,1088.28 988.536,1079.45 1000.5,1072C 1074,1069.95 1147.34,1066.45 1220.5,1061.5C 1221.61,1058.84 1221.61,1056.18 1220.5,1053.5C 1223.27,1050.49 1222.61,1048.49 1218.5,1047.5C 1218.86,1045.09 1217.86,1043.59 1215.5,1043C 1216.79,1040.91 1216.12,1039.58 1213.5,1039C 1213.83,1038.67 1214.17,1038.33 1214.5,1038C 1213.01,1035.38 1212.34,1032.55 1212.5,1029.5C 1218.28,1038.74 1223.78,1048.08 1229,1057.5C 1231.4,1062.49 1229.9,1065.99 1224.5,1068C 1221.44,1068.26 1218.44,1068.76 1215.5,1069.5C 1213.83,1068.83 1212.17,1068.17 1210.5,1067.5C 1180.44,1069.63 1150.44,1071.63 1120.5,1073.5C 1110.81,1073.47 1101.15,1073.47 1091.5,1073.5C 1059.05,1075.76 1026.71,1077.6 994.5,1079C 913.217,1150.34 818.05,1186.51 709,1187.5C 599.827,1186.84 504.661,1150.84 423.5,1079.5C 400.74,1077.32 377.74,1076.15 354.5,1076C 336.48,1074.62 318.647,1073.45 301,1072.5C 299.699,1072.96 298.366,1073.29 297,1073.5C 267.186,1071.67 237.353,1069.67 207.5,1067.5C 201.897,1069.4 196.231,1069.24 190.5,1067C 188.406,1065.74 187.406,1063.91 187.5,1061.5C 188.453,1057.6 189.953,1053.93 192,1050.5C 227.124,988.791 261.957,926.791 296.5,864.5C 271.3,753.389 287.133,648.723 344,550.5C 348.667,543.167 353.333,535.833 358,528.5C 362.418,522.603 366.918,516.936 371.5,511.5 Z"/></g>
|
||||
<g><path style="opacity:1" fill="#4a4730" d="M 788.5,1177.5 C 783.067,1179.12 777.4,1180.12 771.5,1180.5C 772.952,1179.7 774.618,1179.2 776.5,1179C 774.5,1178.67 772.5,1178.33 770.5,1178C 776.486,1176.39 782.486,1176.22 788.5,1177.5 Z"/></g>
|
||||
</svg>
|
After Width: | Height: | Size: 12 KiB |
245
repos_extra/frontpage/readme.md
Normal file
245
repos_extra/frontpage/readme.md
Normal file
@@ -0,0 +1,245 @@
|
||||
# frontpage
|
||||
|
||||
- https://frontpage.fyi
|
||||
- https://bsky.app/profile/frontpage.fyi
|
||||
- https://github.com/likeandscribe/frontpage
|
||||
|
||||
```sh
|
||||
$ git clone https://github.com/likeandscribe/frontpage
|
||||
$ dir=${0:a:h}/frontpage
|
||||
$ cd $dir
|
||||
```
|
||||
|
||||
## first setting
|
||||
|
||||
- https://o.syui.ai
|
||||
- https://at.syu.is/at/syui.ai/ai.syui.o.post
|
||||
|
||||
```sh
|
||||
$ cd $dir
|
||||
$ nvm use 20
|
||||
$ pnpm i
|
||||
$ cat turbo.json
|
||||
$ pnpm exec turbo run --affected type-check
|
||||
```
|
||||
|
||||
```sh
|
||||
$ cd $dir/packages/frontpage
|
||||
$ pnpm exec tsx ./scripts/generate-jwk.mts
|
||||
# pnpm run db:generate
|
||||
# pnpm run db:migrate
|
||||
$ cat .env.local
|
||||
```
|
||||
|
||||
```sh
|
||||
# frontpage/.env.local
|
||||
PRIVATE_JWK=`pnpm exec tsx ./scripts/generate-jwk.mts`
|
||||
PUBLIC_JWK=`pnpm exec tsx ./scripts/generate-jwk.mts`
|
||||
|
||||
TURSO_CONNECTION_URL=`turso db create` #libsql://xxx.turso.io
|
||||
#TURSO_CONNECTION_URL=`turso db shell xxx-xxx`
|
||||
TURSO_AUTH_TOKEN=`turso db tokens create xxx-xxx`
|
||||
|
||||
DRAINPIPE_CONSUMER_SECRET=`openssl ecparam --name secp256k1 --genkey --noout --outform DER | tail --bytes=+8 | head --bytes=32 | xxd --plain --cols 32`
|
||||
VERCEL_PROJECT_PRODUCTION_URL=example.com
|
||||
VERCEL_BRANCH_URL=example.com
|
||||
|
||||
#DRAINPIPE_CONSUMER_SECRET=secret
|
||||
#TURSO_CONNECTION_URL=libsql://turso.dev.unravel.fyi
|
||||
#PLC_DIRECTORY_URL=https://plc.dev.unravel.fyi
|
||||
```
|
||||
|
||||
```sh
|
||||
$ cd $dir/packages-rs/drainpipe
|
||||
$ cargo install diesel_cli --no-default-features --features sqlite
|
||||
$ diesel setup
|
||||
$ diesel migration run
|
||||
$ ls drainpipe.db
|
||||
$ cat .env.local
|
||||
```
|
||||
|
||||
```sh
|
||||
# drainpipe/.env.local
|
||||
DATABASE_URL="drainpipe.db"
|
||||
FRONTPAGE_CONSUMER_URL="http://${cloudflared}/api/receive_hook"
|
||||
FRONTPAGE_CONSUMER_SECRET=`openssl ecparam --name secp256k1 --genkey --noout --outform DER | tail --bytes=+8 | head --bytes=32 | xxd --plain --cols 32`
|
||||
#RELAY_URL=wss://bsky.network
|
||||
#FRONTPAGE_CONSUMER_SECRET=secret
|
||||
```
|
||||
|
||||
## rewrite
|
||||
|
||||
```sh
|
||||
$ cd $dir/packages/frontpage
|
||||
$ PUBLIC_URL=example.com
|
||||
$ grep -R frontpage.fyi ./app ./lib |cut -d : -f 1|xargs sed -i "s/frontpage.fyi/${PUBLIC_URL}/g"
|
||||
|
||||
$ HOST_REVERT=com.unravel.example
|
||||
$ grep -R unravel.frontpage ./app ./lib |cut -d : -f 1|xargs sed -i "s/fyi.unravel.frontpage/${HOST_REVERT}/g"
|
||||
```
|
||||
|
||||
```sh
|
||||
$ cd $dir/packages-rs/drainpipe
|
||||
$ HOST_REVERT=com.unravel.example
|
||||
$ grep -R fyi.unravel.frontpage ./src |cut -d : -f 1|xargs sed -i "s/fyi.unravel.frontpage/${HOST_REVERT}/g"
|
||||
```
|
||||
|
||||
## deploy
|
||||
|
||||
```sh
|
||||
$ cd $dir/packages-rs/drainpipe
|
||||
$ docker compose up
|
||||
---
|
||||
$ cd $dir/packages/frontpage
|
||||
$ docker compose up
|
||||
```
|
||||
|
||||
## explanation
|
||||
|
||||
### client-metadata.json
|
||||
|
||||
the `client_id` is different between `pnpm run start` and `pnpm run dev`. see `https://localhost:3000/oauth/client-metadata.json` for this.
|
||||
|
||||
### local-infra
|
||||
|
||||
i think this is the server configuration required for self-hosting.
|
||||
|
||||
https://github.com/likeandscribe/frontpage/tree/main/packages/frontpage/local-infra
|
||||
|
||||
since plc gives an error, do the following. probably a postgres database is required. there is no need to open ports.
|
||||
|
||||
```yml
|
||||
plc:
|
||||
image: ghcr.io/bluesky-social/did-method-plc:plc-f2ab7516bac5bc0f3f86842fa94e996bd1b3815b
|
||||
container_name: plc
|
||||
restart: unless-stopped
|
||||
ports:
|
||||
- '4000:8080'
|
||||
depends_on:
|
||||
- plc_db
|
||||
env_file:
|
||||
- ./plc.env
|
||||
|
||||
plc_db:
|
||||
image: postgres:16-alpine
|
||||
restart: always
|
||||
env_file:
|
||||
- ./postgres.env
|
||||
volumes:
|
||||
- ./configs/postgres/init/:/docker-entrypoint-initdb.d/
|
||||
- ./data/postgres/:/var/lib/postgresql/data/
|
||||
healthcheck:
|
||||
test: "pg_isready -U $$POSTGRES_USER -d $$POSTGRES_DB"
|
||||
interval: 5s
|
||||
retries: 20
|
||||
```
|
||||
|
||||
```sh
|
||||
# plc.env
|
||||
DEBUG_MODE=1
|
||||
LOG_ENABLED=true
|
||||
LOG_LEVEL=debug
|
||||
LOG_DESTINATION=1
|
||||
PORT=8080
|
||||
DATABASE_URL=postgres://postgres:postgres@plc_db/plc
|
||||
DB_CREDS_JSON='{"username":"postgres","password":"postgres","host":"plc_db","port":"5432","database":"plc"}'
|
||||
ENABLE_MIGRATIONS=true
|
||||
DB_MIGRATE_CREDS_JSON='{"username":"postgres","password":"postgres","host":"plc_db","port":"5432","database":"plc"}'
|
||||
```
|
||||
|
||||
```sh
|
||||
# configs/postgres/init/init.sql
|
||||
-- PLC
|
||||
CREATE DATABASE plc;
|
||||
GRANT ALL PRIVILEGES ON DATABASE plc TO postgres;
|
||||
```
|
||||
|
||||
### pds
|
||||
|
||||
first, i think you need to get the pdsurl with oauth(session). if you have a session, you can perform operations such as posting.
|
||||
|
||||
```sh
|
||||
$ cd $dir/packages/frontpage
|
||||
./lib/data/user.ts: const pdsUrl = await getPdsUrl(session.user.did);
|
||||
```
|
||||
|
||||
it seems that drainpipe searches for `fyi.unravel.frontpage(collection)` in pds and commits it to firehose subscriberepos. if you change these two parts, it will not work with `frontpage.fyi`.
|
||||
|
||||
```rust
|
||||
// https://github.com/likeandscribe/frontpage/blob/e7444ec6c19f0ccef3776f04702c3bb033ed3bfc/packages-rs/drainpipe/src/main.rs#L66-L97
|
||||
// RELAY_URL=wss://bsky.network
|
||||
let mut ws_request = format!(
|
||||
"{}/xrpc/com.atproto.sync.subscribeRepos{}",
|
||||
relay_url, query_string
|
||||
)
|
||||
|
||||
/// Process a message from the firehose. Returns the sequence number of the message or an error.
|
||||
async fn process(message: Vec<u8>, ctx: &mut Context) -> Result<i64, ProcessError> {
|
||||
let (_header, data) = firehose::read(&message).map_err(|e| ProcessError {
|
||||
inner: e.into(),
|
||||
seq: -1,
|
||||
source: message.clone().into(),
|
||||
kind: ProcessErrorKind::DecodeError,
|
||||
})?;
|
||||
let sequence = match data {
|
||||
firehose::SubscribeRepos::Commit(commit) => {
|
||||
let frontpage_ops = commit
|
||||
.operations
|
||||
.iter()
|
||||
.filter(|op| op.path.starts_with("com.unravel.example."))
|
||||
//.filter(|op| op.path.starts_with("fyi.unravel.frontpage."))
|
||||
.collect::<Vec<_>>();
|
||||
if !frontpage_ops.is_empty() {
|
||||
process_frontpage_ops(&frontpage_ops, &commit, &ctx)
|
||||
.map_err(|e| ProcessError {
|
||||
seq: commit.sequence,
|
||||
inner: e,
|
||||
source: message.clone().into(),
|
||||
kind: ProcessErrorKind::ProcessError,
|
||||
})
|
||||
.await?;
|
||||
}
|
||||
commit.sequence
|
||||
}
|
||||
msg => msg.sequence(),
|
||||
};
|
||||
|
||||
Ok(sequence)
|
||||
}
|
||||
```
|
||||
|
||||
色を変更するには`packages/frontpage/app/globals.css `を変更します。
|
||||
|
||||
```css
|
||||
--primary: 210 100% 50%; /* HSLカラーで青色を定義 */
|
||||
--primary-foreground: 0 0% 100%; /* 白色のテキスト */
|
||||
```
|
||||
|
||||
## other
|
||||
|
||||
### delete record
|
||||
|
||||
```sh
|
||||
# at://did:plc:uqzpqmrjnptsxezjx4xuh2mn/ai.syui.o.post/3lafos6to6y2y
|
||||
$ git clone https://git.syui.ai/ai/bot
|
||||
$ cargo build
|
||||
$ ./target/debug/ai delete 3lafos6to6y2y -c ai.syui.o.post
|
||||
```
|
||||
|
||||
### license view
|
||||
|
||||
```html
|
||||
// https://github.com/likeandscribe/frontpage/blob/de31aedf73c4e80e7376cf73c7c054437563f2ab/packages/frontpage/app/layout.tsx#L28-L52
|
||||
+ <div dangerouslySetInnerHTML={{__html: '<!-- frontpage | MIT | https://github.com/likeandscribe/frontpage/blob/main/LICENSE -->'}} />
|
||||
```
|
||||
|
||||
### admin view
|
||||
|
||||
```sh
|
||||
# https://github.com/likeandscribe/frontpage/blob/7fccf20fa800ba25fd57db279033ddf2cc92e9ce/packages/frontpage/lib/constants.ts
|
||||
./lib/constants.ts:export const FRONTPAGE_ATPROTO_HANDLE = "admin.example.com";
|
||||
|
||||
# https://github.com/likeandscribe/frontpage/blob/cf8a4cb8bc7bab54407972964f8d39bf5e7c9182/packages/frontpage/app/(app)/layout.tsx#L55-L66
|
||||
./app/\(app\)/layout.tsx:@admin.example.com <OpenInNewWindowIcon className="inline" />
|
||||
```
|
||||
|
11
repos_extra/pdsls/Dockerfile
Normal file
11
repos_extra/pdsls/Dockerfile
Normal file
@@ -0,0 +1,11 @@
|
||||
FROM node:20
|
||||
|
||||
RUN npm install -g pnpm
|
||||
WORKDIR /app
|
||||
RUN git clone https://github.com/notjuliet/pdsls
|
||||
WORKDIR /app/pdsls
|
||||
COPY ./vite.config.ts ./vite.config.ts
|
||||
RUN pnpm i
|
||||
RUN pnpm build
|
||||
|
||||
CMD [ "pnpm", "start"]
|
6
repos_extra/pdsls/compose.yml
Normal file
6
repos_extra/pdsls/compose.yml
Normal file
@@ -0,0 +1,6 @@
|
||||
services:
|
||||
atbrowser:
|
||||
build:
|
||||
context: .
|
||||
ports:
|
||||
- 3000:13213
|
17
repos_extra/pdsls/vite.config.ts
Normal file
17
repos_extra/pdsls/vite.config.ts
Normal file
@@ -0,0 +1,17 @@
|
||||
import { defineConfig } from "vite";
|
||||
import solidPlugin from "vite-plugin-solid";
|
||||
import UnoCSS from "unocss/vite";
|
||||
|
||||
const SERVER_HOST = "0.0.0.0";
|
||||
const SERVER_PORT = 13213;
|
||||
|
||||
export default defineConfig({
|
||||
plugins: [UnoCSS(), solidPlugin()],
|
||||
server: {
|
||||
host: SERVER_HOST,
|
||||
port: SERVER_PORT,
|
||||
},
|
||||
build: {
|
||||
target: "esnext",
|
||||
},
|
||||
});
|
7
repos_extra/python-oauth-web-app/Dockerfile
Normal file
7
repos_extra/python-oauth-web-app/Dockerfile
Normal file
@@ -0,0 +1,7 @@
|
||||
FROM syui/aios
|
||||
|
||||
WORKDIR /app
|
||||
COPY . .
|
||||
RUN pacman -Syu rye --noconfirm
|
||||
RUN rye sync
|
||||
|
3
repos_extra/python-oauth-web-app/app_add.py
Normal file
3
repos_extra/python-oauth-web-app/app_add.py
Normal file
@@ -0,0 +1,3 @@
|
||||
@app.route("/about")
|
||||
def aboutpage():
|
||||
return render_template("about.html")
|
10
repos_extra/python-oauth-web-app/compose.yml
Normal file
10
repos_extra/python-oauth-web-app/compose.yml
Normal file
@@ -0,0 +1,10 @@
|
||||
services:
|
||||
web:
|
||||
build: .
|
||||
env_file:
|
||||
- ./.env
|
||||
ports:
|
||||
- "5000:5000"
|
||||
volumes:
|
||||
- ./demo.sqlite:/app/demo.sqlite
|
||||
command: rye run flask run --host=0.0.0.0
|
20
repos_extra/python-oauth-web-app/templates/about.html
Normal file
20
repos_extra/python-oauth-web-app/templates/about.html
Normal file
@@ -0,0 +1,20 @@
|
||||
{% extends 'base.html' %}
|
||||
{% block content %}
|
||||
|
||||
<h2>service</h2>
|
||||
<p>This service allows you to comment on live broadcasts using your atproto account.</p>
|
||||
|
||||
<h2>system</h2>
|
||||
<p>Display posts to BBS using atproto oauth.</p>
|
||||
<p>This service is generated using <a href="https://github.com/bluesky-social/cookbook">bluesky/cookbook</a>.</p>
|
||||
<p>Authentication information will be deleted periodically.</p>
|
||||
<hr/>
|
||||
|
||||
<h2>サービス</h2>
|
||||
<p>このサービスはlive配信にatprotoアカウントを使ってコメントができます。</p>
|
||||
<h2>システム</h2>
|
||||
<p>atproto oauthを使用してbbsへの書き込みを許可します。</p>
|
||||
<p>このサービスは<a href="https://github.com/bluesky-social/cookbook">bluesky/cookbook</a>を使用して生成されています。</p>
|
||||
<p>認証情報は定期的に削除されます。</p>
|
||||
|
||||
{% endblock %}
|
114
repos_extra/python-oauth-web-app/templates/base.html
Normal file
114
repos_extra/python-oauth-web-app/templates/base.html
Normal file
@@ -0,0 +1,114 @@
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="referrer" content="origin-when-cross-origin">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@picocss/pico@2/css/pico.classless.blue.min.css">
|
||||
<link rel="stylesheet" href="https://syui.ai/bower_components/font-awesome/css/all.min.css" />
|
||||
<link rel="icon" href="https://live.syui.ai/favicon.ico">
|
||||
<title>o.syui.ai</title>
|
||||
</head>
|
||||
<body>
|
||||
<header>
|
||||
<hgroup>
|
||||
{% if g.user %}
|
||||
{% endif %}
|
||||
</hgroup>
|
||||
<nav>
|
||||
<ul>
|
||||
<li><a href="/"><button class="secondary">o</button></a></li>
|
||||
<li>.syui.ai</li>
|
||||
</ul>
|
||||
<ul>
|
||||
{% if g.user %}
|
||||
<li><a href="{{ url_for('bsky_post') }}">post</a>
|
||||
<li><a href="{{ url_for('oauth_refresh') }}">refresh</a>
|
||||
<li><a href="{{ url_for('oauth_logout') }}"><button class="secondary">logout</button></a>
|
||||
{% else %}
|
||||
<li><a href="/about"><i class="fa-solid fa-circle-question"></i></a></li>
|
||||
{% endif %}
|
||||
<!--
|
||||
<li><a href="https://github.com/bluesky-social/cookbook">Code</a>
|
||||
-->
|
||||
</ul>
|
||||
</nav>
|
||||
</header>
|
||||
|
||||
<main class="container">
|
||||
<section class="content">
|
||||
{% for message in get_flashed_messages() %}
|
||||
<article>{{ message }}</article>
|
||||
{% endfor %}
|
||||
{% block content %}{% endblock %}
|
||||
</section>
|
||||
</main>
|
||||
|
||||
<style>
|
||||
body {
|
||||
background-color: #f1f1f1;
|
||||
}
|
||||
button a {
|
||||
color: #fff;
|
||||
}
|
||||
iframe#livechat {
|
||||
width: 100%;
|
||||
min-height: 500px;
|
||||
}
|
||||
iframe#livestream {
|
||||
width: 100%;
|
||||
min-height: 500px;
|
||||
}
|
||||
button.oauth-login {
|
||||
width: 100%;
|
||||
}
|
||||
footer {
|
||||
text-align:center;
|
||||
}
|
||||
|
||||
.container-1 {
|
||||
display: flex;
|
||||
.right-container {
|
||||
background-color: #fff;
|
||||
position: sticky;
|
||||
top: 0;
|
||||
align-self: flex-start;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: flex-end;
|
||||
padding:10px;
|
||||
margin-left:30px;
|
||||
.content:first-of-type {
|
||||
position: sticky;
|
||||
bottom: 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.left-container {
|
||||
padding:10px;
|
||||
background-color: #fff;
|
||||
width: 80%;
|
||||
}
|
||||
|
||||
@media screen and (max-width: 959px) {
|
||||
.left-container {
|
||||
width: 100%;
|
||||
}
|
||||
iframe#livestream {
|
||||
width: 100%;
|
||||
min-height:300px;
|
||||
height:100%;
|
||||
}
|
||||
.container-1 {
|
||||
display: block;
|
||||
.right-container {
|
||||
margin-left:0px;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
</style>
|
||||
<footer>©syui</footer>
|
||||
</body>
|
||||
</html>
|
@@ -0,0 +1,8 @@
|
||||
{% extends 'base.html' %}
|
||||
|
||||
{% block content %}
|
||||
<form method="post">
|
||||
<textarea name="post_text" placeholder="What's up?" id="post_text" required></textarea>
|
||||
<input type="submit" value="Poast!">
|
||||
</form>
|
||||
{% endblock %}
|
13
repos_extra/python-oauth-web-app/templates/error.html
Normal file
13
repos_extra/python-oauth-web-app/templates/error.html
Normal file
@@ -0,0 +1,13 @@
|
||||
{% extends 'base.html' %}
|
||||
|
||||
{% block title %}Error{% endblock %}
|
||||
|
||||
{% block content %}
|
||||
<h2>⚠️ Error {{ status_code }} ⚠️</h2>
|
||||
{% if err.description %}
|
||||
<p><code>{{ err.description }}</code></p>
|
||||
{% else %}
|
||||
<p>Something went wrong!</p>
|
||||
{% endif %}
|
||||
<p><a href="/">Start Over</a></p>
|
||||
{% endblock %}
|
BIN
repos_extra/python-oauth-web-app/templates/favicon.png
Normal file
BIN
repos_extra/python-oauth-web-app/templates/favicon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 14 KiB |
1
repos_extra/python-oauth-web-app/templates/favicon.svg
Normal file
1
repos_extra/python-oauth-web-app/templates/favicon.svg
Normal file
@@ -0,0 +1 @@
|
||||
<?xml version="1.0" ?><svg height="512" viewBox="0 0 512 512" width="512" xmlns="http://www.w3.org/2000/svg"><title/><path d="M413.48,284.46c58.87,47.24,91.61,89,80.31,108.55-17.85,30.85-138.78-5.48-270.1-81.15S.37,149.84,18.21,119c11.16-19.28,62.58-12.32,131.64,14.09" style="fill:none;stroke:#000;stroke-miterlimit:10;stroke-width:32px"/><circle cx="256" cy="256" r="160" style="fill:none;stroke:#000;stroke-miterlimit:10;stroke-width:32px"/></svg>
|
After Width: | Height: | Size: 450 B |
29
repos_extra/python-oauth-web-app/templates/home.html
Normal file
29
repos_extra/python-oauth-web-app/templates/home.html
Normal file
@@ -0,0 +1,29 @@
|
||||
{% extends 'base.html' %}
|
||||
{% block content %}
|
||||
|
||||
<div class="container-1">
|
||||
<div class="left-container">
|
||||
<div class="livestream">
|
||||
<iframe id="livestream" title="stream" src="https://live.syui.ai/b9ec42d4-8a4d-4343-99fc-1bd1cdbc5a6f.html"> </iframe>
|
||||
</div>
|
||||
</div>
|
||||
<div class="right-container">
|
||||
{% if g.user %}
|
||||
<span class="user-handle">@{{ g.user['handle'] }}</span>
|
||||
<div class="livechat">
|
||||
<iframe id="livechat" title="bskychat" src="https://bbs.syui.ai/?handle={{ session['user_handle'] }}"> </iframe>
|
||||
</div>
|
||||
{% else %}
|
||||
<div class="livechat">
|
||||
<a href="{{ url_for('oauth_login') }}"><button class="oauth-login">@</button></a>
|
||||
<iframe id="livechat" title="bskychat" src="https://bbs.syui.ai"> </iframe>
|
||||
</div>
|
||||
<div class="right-container">
|
||||
<p>write comment using oauth atproto.</p>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
{% endblock %}
|
15
repos_extra/python-oauth-web-app/templates/login.html
Normal file
15
repos_extra/python-oauth-web-app/templates/login.html
Normal file
@@ -0,0 +1,15 @@
|
||||
{% extends 'base.html' %}
|
||||
|
||||
{% block content %}
|
||||
<article>
|
||||
<h3>Login with atproto</h3>
|
||||
<form method="post">
|
||||
<p>Provide your handle or DID to authorize an existing account with PDS.
|
||||
<br>You can also supply a PDS/entryway URL (eg, <code>https://pds.example.com</code>).</p>
|
||||
<fieldset role="group">
|
||||
<input name="username" id="username" placeholder="handle.example.com" style="font-family: monospace,monospace;" required>
|
||||
<input type="submit" value="Login">
|
||||
</fieldset>
|
||||
</form>
|
||||
</article>
|
||||
{% endblock %}
|
5
repos_extra/readme.md
Normal file
5
repos_extra/readme.md
Normal file
@@ -0,0 +1,5 @@
|
||||
https://github.com/notjuliet/pdsls
|
||||
|
||||
https://github.com/likeandscribe/frontpage
|
||||
|
||||
https://github.com/bluesky-social/cookbook/tree/main/python-oauth-web-app
|
60
scpt/at.zsh
Executable file
60
scpt/at.zsh
Executable file
@@ -0,0 +1,60 @@
|
||||
#!/bin/zsh
|
||||
|
||||
d=${0:a:h}
|
||||
|
||||
source $d/src/tag.zsh
|
||||
source $d/src/func.zsh
|
||||
|
||||
at-env
|
||||
case $1 in
|
||||
version|v)
|
||||
at-version
|
||||
;;
|
||||
uri|u)
|
||||
at-uri $2
|
||||
;;
|
||||
did|d)
|
||||
at-did $2
|
||||
;;
|
||||
collection|c)
|
||||
at-collection $2 $3
|
||||
;;
|
||||
at-docs|docs)
|
||||
at-docs
|
||||
;;
|
||||
cid)
|
||||
at-cid $2 $3
|
||||
;;
|
||||
login|l)
|
||||
at-login $2 $3
|
||||
;;
|
||||
refresh|r)
|
||||
at-refresh
|
||||
;;
|
||||
session|s)
|
||||
at-session
|
||||
;;
|
||||
profile|pro)
|
||||
at-profile
|
||||
;;
|
||||
download-character-icon)
|
||||
source $d/ai/syui/game/character.zsh
|
||||
download_character_icon
|
||||
;;
|
||||
create-game-character)
|
||||
source $d/ai/syui/game/character.zsh
|
||||
create_game_character
|
||||
;;
|
||||
ai-create-user)
|
||||
source $d/ai/syui/game/character.zsh
|
||||
ai_create_user
|
||||
;;
|
||||
*)
|
||||
echo "${help[@]}"
|
||||
echo "${host[@]}"
|
||||
echo "${lexicon[@]}"
|
||||
echo "${github[@]}"
|
||||
echo "${tag[@]}"
|
||||
;;
|
||||
esac
|
||||
at-unset
|
302
scpt/src/ai/syui/game/character.zsh
Executable file
302
scpt/src/ai/syui/game/character.zsh
Executable file
@@ -0,0 +1,302 @@
|
||||
character=(
|
||||
ai
|
||||
manny
|
||||
quinn
|
||||
chinese
|
||||
kirin
|
||||
leviathan
|
||||
phoenix
|
||||
wyvern
|
||||
cerberus
|
||||
dragon
|
||||
kitsune
|
||||
pegasus
|
||||
)
|
||||
|
||||
img=(
|
||||
bafkreie34pjuc6coenzcdwrgrh4fbacq7bkhsz263g5vpbsqxwaz37kkwy
|
||||
bafkreie34pjuc6coenzcdwrgrh4fbacq7bkhsz263g5vpbsqxwaz37kkwy
|
||||
bafkreie34pjuc6coenzcdwrgrh4fbacq7bkhsz263g5vpbsqxwaz37kkwy
|
||||
bafkreidlealfybajqzwv5eoz4jshnsijc2vnktlhpw4ph47krwj6aigqby
|
||||
bafkreiegpqedlrfa4ljhssdnkrr5hyd5huy2xhh2zszj5wq2wuuzejggmq
|
||||
bafkreig6vszkx3c4dcortjwfsz6sa6zwqgj7zpxj4lxfrrkwql4xhiu5ou
|
||||
bafkreich7fsumke2yvumvixkruonzrcevk3f6g2cntzfwdn4n2c2vox5dm
|
||||
bafkreiacjvagsekhiiljz3j237b6klrt6pkptxljt7kltprgg5276gv25q
|
||||
bafkreihpni4lp55jysalcntulzal5rbhidtbseanlucpyucagzxmv6xj24
|
||||
bafkreia3huw2gdenqatoobx3hcft74chced46bw4znfgepo5aenegobkri
|
||||
bafkreidy74aieb6ie646xhosginox5zbnbnrtd76cnt4pbn73hrxgfnple
|
||||
bafkreifnbfj27fr6nv7qeqqmwdibf7qrw4lauvzoknw5hexbifmwqt6kmq
|
||||
)
|
||||
|
||||
function download_character_icon(){
|
||||
|
||||
t=(
|
||||
"https://sketchfab.com/3d-models/super-9a80a6d6cf6f4b08906505c7f945d3ce"
|
||||
)
|
||||
|
||||
t=(
|
||||
"https://sketchfab.com/3d-models/cerberus-quirky-series-4379b571b5a440119d1ebaddb0711142"
|
||||
"https://sketchfab.com/3d-models/chinese-dragon-quirky-series-a383d3cf5b004978ac620806558b2924"
|
||||
"https://sketchfab.com/3d-models/dragon-quirky-series-9a0989aae9b84ebdade28e84a0702a71"
|
||||
"https://sketchfab.com/3d-models/kirin-quirky-series-b280c8bc5b87471eac1068acc91fdce1"
|
||||
"https://sketchfab.com/3d-models/kitsune-quirky-series-4fc8b2ade43f4d4bb8a8e6e227f00a62"
|
||||
"https://sketchfab.com/3d-models/leviathan-quirky-series-002200e1db2c461fbcaa8d2fdac2d766"
|
||||
"https://sketchfab.com/3d-models/pegasus-quirky-series-a4488ae7a2d2405c927a50f5a8b2d6bb"
|
||||
"https://sketchfab.com/3d-models/phoenix-quirky-series-1f0a01247b78441ab5b9cf8e9711e78e"
|
||||
"https://sketchfab.com/3d-models/wyvern-quirky-series-7baad217325a45b4877514b3f5924be9"
|
||||
)
|
||||
|
||||
for i in $t; do
|
||||
name=`echo $i|cut -d / -f 5|cut -d - -f 1`
|
||||
tt=`curl -sL $i|tr ' ' '\n' |grep .jpeg|cut -d '"' -f 2`
|
||||
normal=`echo $tt|awk "NR==1"`
|
||||
min=`echo $tt|awk "NR==2"`
|
||||
if [ ! -f $name.jpeg ];then
|
||||
curl -sL $normal -o $name.jpeg
|
||||
fi
|
||||
if [ ! -f ${name}-min.jpeg ];then
|
||||
#curl -sL $min -o ${name}-min.jpeg
|
||||
fi
|
||||
array+=(`echo $tt|sed "1,2d"|cut -d ";" -f 2|cut -d '&' -f 1|tr '\n' ' '`)
|
||||
echo $name
|
||||
for ((i=1; i<=$#array; i++)); do
|
||||
echo "Index: $i, Value: ${array[$i]}"
|
||||
#curl -sL ${array[$i]} -o ${name}-${i}.jpeg
|
||||
done
|
||||
done
|
||||
}
|
||||
|
||||
|
||||
|
||||
case $OSTYPE in
|
||||
darwin*)
|
||||
day=`gdate --iso-8601=seconds`
|
||||
;;
|
||||
*)
|
||||
day=`date --iso-8601=seconds`
|
||||
;;
|
||||
esac
|
||||
|
||||
function create_game_character() {
|
||||
t=(
|
||||
ai
|
||||
chinese
|
||||
kirin
|
||||
leviathan
|
||||
phoenix
|
||||
wyvern
|
||||
cerberus
|
||||
dragon
|
||||
kitsune
|
||||
pegasus
|
||||
)
|
||||
|
||||
for ((i=1; i<=$#t; i++)); do
|
||||
created=2020-01-01T00:00:00+09:00
|
||||
col=ai.syui.game.character
|
||||
req=com.atproto.repo.getRecord
|
||||
url=https://$host/xrpc/$req
|
||||
|
||||
id=$i
|
||||
name=${t[$i]}
|
||||
chara=$name
|
||||
rkey=$chara
|
||||
repo=$did_yui
|
||||
json="{\"collection\":\"$col\", \"rkey\":\"$rkey\", \"repo\":\"$repo\"}"
|
||||
if [ $((RANDOM % 2)) -eq 0 ];then
|
||||
gender=male
|
||||
else
|
||||
gender=female
|
||||
fi
|
||||
|
||||
case $name in
|
||||
ai)
|
||||
gender=none
|
||||
group=origin
|
||||
season=0
|
||||
;;
|
||||
chinese|dragon|cerberus|pegasus|leviathan)
|
||||
gender=male
|
||||
group=fantasy
|
||||
season=1
|
||||
;;
|
||||
kitsune|phoenix|kirin|wyvern)
|
||||
gender=female
|
||||
group=fantasy
|
||||
season=1
|
||||
;;
|
||||
*)
|
||||
continue ;;
|
||||
esac
|
||||
|
||||
jj=`curl -sL "$url?repo=$repo&collection=$col&rkey=$rkey"`
|
||||
|
||||
link=`echo $jj|jq -r '.value.embed.external.thumb.ref.[]'`
|
||||
size=`echo $jj|jq -r .value.embed.external.thumb.size`
|
||||
mtype=`echo $jj|jq -r .value.embed.external.thumb.mimeType`
|
||||
echo $name
|
||||
echo $gender
|
||||
echo https://cdn.bsky.app/img/feed_thumbnail/plain/$did_yui/$link
|
||||
|
||||
## upload img
|
||||
#if [ -f ./${name}.jpeg ];then
|
||||
# jj=`ai img-upload ./${name}.jpeg`
|
||||
#elif [ -f ./${name}.png ];then
|
||||
# jj=`ai img-upload ./${name}.png`
|
||||
#fi
|
||||
#link=`echo $jj|jq -r ".blob.ref.[]"`
|
||||
#size=`echo $jj|jq -r .blob.size`
|
||||
#mtype=`echo $jj|jq -r .blob.mimeType`
|
||||
|
||||
req=com.atproto.repo.putRecord
|
||||
url=https://$host/xrpc/$req
|
||||
|
||||
nickname=$name
|
||||
fullname=$name
|
||||
uri=at://${did_yui}/$col/$chara
|
||||
|
||||
|
||||
json="{
|
||||
\"repo\": \"$handle_yui\",
|
||||
\"did\": \"$did_yui\",
|
||||
\"collection\": \"$col\",
|
||||
\"rkey\": \"$chara\",
|
||||
\"record\": {
|
||||
\"id\": $id,
|
||||
\"name\": \"$name\",
|
||||
\"fullname\": \"$fullname\",
|
||||
\"nickname\": \"$nickname\",
|
||||
\"gender\": \"$gender\",
|
||||
\"season\": $season,
|
||||
\"group\": \"$group\",
|
||||
\"embed\": {
|
||||
\"\$type\": \"app.bsky.embed.external\",
|
||||
\"external\": {
|
||||
\"uri\": \"$uri\",
|
||||
\"thumb\": {
|
||||
\"\$type\": \"blob\",
|
||||
\"ref\": {
|
||||
\"\$link\": \"$link\"
|
||||
},
|
||||
\"mimeType\": \"$mtype\",
|
||||
\"size\": $size
|
||||
} } }, \"createdAt\": \"$created\", \"updatedAt\": \"$day\" } }"
|
||||
|
||||
if echo $json|jq . ;then
|
||||
curl -sL -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $token_yui" -d $json $url
|
||||
fi
|
||||
done
|
||||
|
||||
}
|
||||
|
||||
function ai_create_user() {
|
||||
|
||||
col=ai.syui.game.user
|
||||
rkey=ai
|
||||
did=$did_yui
|
||||
handle=$handle_yui
|
||||
#rkey=syui
|
||||
#img=https://cdn.bsky.app/img/feed_thumbnail/plain/did:plc:4hqjfn7m6n5hno3doamuhgef/bafkreie34pjuc6coenzcdwrgrh4fbacq7bkhsz263g5vpbsqxwaz37kkwy@jpeg
|
||||
req=com.atproto.repo.putRecord
|
||||
url=https://$host/xrpc/$req
|
||||
|
||||
for ((i=1; i<=${#character}; i++)); do
|
||||
|
||||
cname=${character[$i]}
|
||||
cid=${img[$i]}
|
||||
imguri=https://cdn.bsky.app/img/feed_thumbnail/plain/$did_yui/${cid}@jpeg
|
||||
echo $imguri
|
||||
echo $i $character $cname
|
||||
season=1
|
||||
group=test
|
||||
lv=1
|
||||
hp=100
|
||||
exp=0
|
||||
rank=0
|
||||
mode=0
|
||||
attack=0
|
||||
attack_post=0
|
||||
critical=0
|
||||
critical_d=0
|
||||
|
||||
case $cname in
|
||||
ai)
|
||||
group=origin
|
||||
season=0
|
||||
mode=2
|
||||
;;
|
||||
manny|quinn)
|
||||
group=test
|
||||
;;
|
||||
*)
|
||||
group=fantasy
|
||||
;;
|
||||
esac
|
||||
|
||||
jc="\"$cname\": { \"group\": \"$group\", \"season\": $season, \"img\": \"$imguri\", \"lv\": $lv, \"exp\": $exp, \"hp\": $hp, \"rank\": $rank, \"mode\": $mode, \"attack\": $attack, \"attack_post\": $attack_post, \"critical\": $critical, \"critical_d\": $critical_d }"
|
||||
|
||||
if [ $#character -ne $i ];then
|
||||
jc=${jc},
|
||||
fi
|
||||
|
||||
json=${json}${jc}
|
||||
|
||||
done
|
||||
json="
|
||||
{
|
||||
\"repo\": \"$handle_yui\",
|
||||
\"did\": \"$did_yui\",
|
||||
\"collection\": \"$col\",
|
||||
\"rkey\": \"$rkey\",
|
||||
\"record\": {
|
||||
\"did\": \"$did\",
|
||||
\"limit\": false,
|
||||
\"login\": false,
|
||||
\"handle\": \"$handle\",
|
||||
\"aiten\": 10,
|
||||
\"gender\": \"famale\",
|
||||
\"character\": {
|
||||
$json
|
||||
},
|
||||
\"createdAt\": \"${created}\",
|
||||
\"updatedAt\": \"${created}\"
|
||||
}
|
||||
}"
|
||||
|
||||
|
||||
if echo $json|jq . ;then
|
||||
echo $token_yui
|
||||
echo y
|
||||
read y
|
||||
if [ "y" = "$y" ];then
|
||||
curl -sL -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $token_yui" -d $json $url
|
||||
fi
|
||||
fi
|
||||
|
||||
col=ai.syui.game
|
||||
rkey=self
|
||||
at_uri=at://$did_yui/ai.syui.game.user/ai
|
||||
|
||||
json="
|
||||
{
|
||||
\"repo\": \"$handle_yui\",
|
||||
\"did\": \"$did_yui\",
|
||||
\"collection\": \"$col\",
|
||||
\"rkey\": \"$rkey\",
|
||||
\"record\": {
|
||||
\"account\": \"$at_uri\",
|
||||
\"username\": \"ai\",
|
||||
\"createdAt\": \"$created\"
|
||||
}
|
||||
}"
|
||||
|
||||
if echo $json|jq . ;then
|
||||
echo y
|
||||
read y
|
||||
if [ "y" = "$y" ];then
|
||||
curl -sL -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $token_yui" -d $json $url
|
||||
fi
|
||||
fi
|
||||
}
|
||||
|
||||
|
136
scpt/src/func.zsh
Normal file
136
scpt/src/func.zsh
Normal file
@@ -0,0 +1,136 @@
|
||||
function at-env() {
|
||||
host=bsky.social
|
||||
handle_syui=syui.ai
|
||||
handle_yui=yui.syui.ai
|
||||
did_yui=did:plc:4hqjfn7m6n5hno3doamuhgef
|
||||
did_syui=did:plc:uqzpqmrjnptsxezjx4xuh2mn
|
||||
handle=$handle_yui
|
||||
did=$did_yui
|
||||
created=2020-01-01T00:00:00+09:00
|
||||
at_uri=at://$did_yui/ai.syui.game.user/syui
|
||||
docs_uri=https://docs.bsky.app/docs/api
|
||||
did=`echo $at_uri|cut -d / -f 3`
|
||||
collection=`echo $at_uri|cut -d / -f 4`
|
||||
rkey=`echo $at_uri|cut -d / -f 5`
|
||||
d=${0:a:h}
|
||||
f=~/.config/.at-zsh.json
|
||||
if [ -f $f ];then
|
||||
token=`cat $f|jq -r .accessJwt`
|
||||
did=`cat $f|jq -r .did`
|
||||
handle=`cat $f|jq -r .handle`
|
||||
actor=$did
|
||||
fi
|
||||
}
|
||||
|
||||
function at-unset() {
|
||||
unset t password token refresh
|
||||
}
|
||||
|
||||
function at-version() {
|
||||
curl -sL ${host}/xrpc/_health |jq .
|
||||
}
|
||||
|
||||
function at-uri() {
|
||||
if [ -n "$1" ];then
|
||||
at_uri=$1
|
||||
fi
|
||||
req=/xrpc/com.atproto.repo.getRecord
|
||||
url=https://${host}${req}
|
||||
did=`echo $at_uri|cut -d / -f 3`
|
||||
collection=`echo $at_uri|cut -d / -f 4`
|
||||
rkey=`echo $at_uri|cut -d / -f 5`
|
||||
curl -sL "$url?repo=$did&collection=$collection&rkey=$rkey"|jq .
|
||||
}
|
||||
|
||||
function at-did() {
|
||||
if [ -n "$1" ];then
|
||||
handle=$1
|
||||
fi
|
||||
req=/xrpc/com.atproto.repo.describeRepo
|
||||
url=https://${host}${req}
|
||||
curl -sL "$url?repo=$handle"|jq .
|
||||
}
|
||||
|
||||
function at-collection() {
|
||||
reverse=false
|
||||
if [ -n "$1" ];then
|
||||
at_uri=$1
|
||||
fi
|
||||
if [ "$2" = "-r" ];then
|
||||
reverse=true
|
||||
fi
|
||||
req=/xrpc/com.atproto.repo.listRecords
|
||||
url=https://${host}${req}
|
||||
did=`echo $at_uri|cut -d / -f 3`
|
||||
collection=`echo $at_uri|cut -d / -f 4`
|
||||
curl -sL "$url?repo=$did&collection=$collection&reverse=$reverse"|jq .
|
||||
}
|
||||
|
||||
function at-docs() {
|
||||
for i in $lexicon; do
|
||||
req=`echo $i|tr '.' '-'|sed 's/[A-Z]/-&/g'|tr '[A-Z]' '[a-z]'`
|
||||
url=$docs_uri/$req
|
||||
echo $url
|
||||
done
|
||||
}
|
||||
|
||||
function at-login() {
|
||||
if [ -z "$1" ] || [ -z "$2" ];then
|
||||
echo handle password
|
||||
exit
|
||||
fi
|
||||
handle=$1
|
||||
password=$2
|
||||
echo $password
|
||||
json="{\"identifier\":\"$handle\",\"password\":\"$password\"}"
|
||||
req=/xrpc/com.atproto.server.createSession
|
||||
url=https://${host}${req}
|
||||
if [ ! -d ~/.config ];then
|
||||
mkdir -p ~/.config
|
||||
fi
|
||||
t=`curl -sL -X POST -H "Content-Type: application/json" -d $json $url`
|
||||
if echo $t |jq .;then
|
||||
echo $t >! $f
|
||||
fi
|
||||
}
|
||||
|
||||
function at-refresh() {
|
||||
echo $f
|
||||
token=`cat $f|jq -r .refreshJwt`
|
||||
req=/xrpc/com.atproto.server.refreshSession
|
||||
url=https://${host}${req}
|
||||
t=`curl -sL -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $token" $url`
|
||||
if echo $t |jq .;then
|
||||
echo $t >! $f
|
||||
fi
|
||||
}
|
||||
|
||||
function at-cid() {
|
||||
did=`at-did $1|jq -r .did`
|
||||
cid=$2
|
||||
req=/xrpc/com.atproto.sync.getBlob
|
||||
url="https://${host}${req}?did=${did}&cid=${cid}"
|
||||
curl -sL $url
|
||||
}
|
||||
|
||||
function at-session() {
|
||||
req=/xrpc/com.atproto.server.getSession
|
||||
url=https://${host}${req}
|
||||
t=`curl -sL -X GET -H "Content-Type: application/json" -H "Authorization: Bearer $token" $url`
|
||||
if ! echo $t |jq .;then
|
||||
echo refresh
|
||||
t=`at-refresh`
|
||||
fi
|
||||
}
|
||||
|
||||
function at-profile() {
|
||||
if [ ! -f $f ];then
|
||||
echo login
|
||||
exit
|
||||
else
|
||||
t=`at-session`
|
||||
fi
|
||||
req=/xrpc/app.bsky.actor.getProfile
|
||||
url="https://${host}${req}?actor=$did"
|
||||
curl -sL -X GET -H "Content-Type: application/json" -H "Authorization: Bearer $token" $url|jq .
|
||||
}
|
39
scpt/src/tag.zsh
Normal file
39
scpt/src/tag.zsh
Normal file
@@ -0,0 +1,39 @@
|
||||
help=(
|
||||
at.zsh v
|
||||
at.zsh d handle
|
||||
at.zsh u at-uri
|
||||
at.zsh c at-uri -r
|
||||
at.zsh docs
|
||||
at.zsh l handle password
|
||||
at.zsh r
|
||||
at.zsh s
|
||||
at.zsh cid handle cid
|
||||
at.zsh pro
|
||||
at.zsh col ai.syui.game.user self
|
||||
)
|
||||
host=(
|
||||
bsky.social
|
||||
public.api.bsky.app
|
||||
plc.directory
|
||||
)
|
||||
lexicon=(
|
||||
# https://github.com/bluesky-social/atproto/tree/main/lexicons
|
||||
com.atproto.repo.describeRepo
|
||||
com.atproto.repo.getRecord
|
||||
com.atproto.repo.listRecords
|
||||
com.atproto.identity.resolveHandle
|
||||
com.atproto.server.createSession
|
||||
com.atproto.server.refreshSession
|
||||
com.atproto.server.getSession
|
||||
com.atproto.sync.getBlob
|
||||
app.bsky.actor.getProfile
|
||||
)
|
||||
github=(
|
||||
https://github.com/bluesky-social/atproto
|
||||
https://github.com/bluesky-social/social-app
|
||||
https://github.com/bluesky-social/feed-generator
|
||||
https://github.com/bluesky-social/jetstream
|
||||
https://github.com/bluesky-social/indigo
|
||||
https://github.com/did-method-plc/did-method-plc
|
||||
)
|
||||
tag=( at bsky bsync pds bgs plc ozone feed jetstream social-app oauth )
|
Reference in New Issue
Block a user