ai/at
1
0

Compare commits

...

21 Commits

Author SHA1 Message Date
ded3110fce update scpt 2025-06-01 01:06:24 +09:00
6f4712e461 add lex 2025-05-27 01:21:35 +09:00
bd3299154c add item 2025-05-12 02:33:03 +09:00
8b97cbfd9b add ozone 2025-05-06 18:53:56 +09:00
246f95a00c fix scpt 2025-05-05 08:35:11 +09:00
cc27f367ea fix patch 2025-04-13 13:00:57 +09:00
72b8f39825 fix scpt 2025-03-26 15:46:26 +09:00
c3d75140d8 fix pdsls 2025-03-18 08:27:55 +09:00
a45ba54323 fix url 2025-03-14 18:37:41 +09:00
a77cdc1e58 add patch 2025-03-12 23:58:00 +09:00
9db0aab05a fix scpt 2025-03-06 13:06:23 +09:00
320d98b2da fix docker 2025-03-05 21:35:33 +09:00
e05868c9b1 fix redirect 2025-03-05 18:53:08 +09:00
9d78ee8627 fix indent 2025-03-01 21:29:25 +09:00
16c785bf0f add patch 2025-03-01 21:22:37 +09:00
b5e44947da update scpt 2025-03-01 20:54:20 +09:00
f26ef781f3 update lexicon 2024-12-27 16:48:54 +09:00
b5b3850f7b update scpt 2024-12-27 16:48:54 +09:00
c61fd5c748 add scpt 2024-12-27 16:48:54 +09:00
005ddc36cf add game 2024-12-27 16:48:54 +09:00
5acaa7aeec update 2024-12-27 16:48:48 +09:00
124 changed files with 4954 additions and 0 deletions

38
.github/workflows/generate-record.yml vendored Normal file
View File

@ -0,0 +1,38 @@
name: Generate Record JSON
on:
workflow_dispatch: # 手動トリガー
push:
paths:
- scpt/generate_record.py # スクリプトに変更があったとき
schedule:
- cron: '0 3 * * *' # 毎日03:00 UTCに自動生成任意
jobs:
build:
runs-on: ubuntu-latest
steps:
- name: Checkout Repository
uses: actions/checkout@v3
- name: Set up Python
uses: actions/setup-python@v5
with:
python-version: '3.x'
- name: Install Dependencies
run: |
pip install -r requirements.txt || true # 必要なら
- name: Run Record Generator Script
run: |
python scpt/generate_record.py
- name: Commit and Push Generated JSON
run: |
git config --global user.name "GitHub Actions Bot"
git config --global user.email "actions@github.com"
git add record.json
git commit -m "🧬 Auto-generate record.json" || echo "No changes to commit"
git push origin main

1
.gitignore vendored Normal file
View File

@ -0,0 +1 @@
repos

128
README.md
View File

@ -0,0 +1,128 @@
# at
https://github.com/bluesky-social/atproto
|word|name|example|
|---|---|---|
|at|uri|at://ai.syu.is|
|@|user|@ai.syu.is|
|[at]proto|repo|`git@github.com:bluesky-social/atproto`|
|[at]mosphere|system|pds, bsky(appview), ozone, bgs, plc|
|[a]uthenticated [t]ransfer|protocol|[did](https://www.w3.org/TR/did-core/)|
- https://atproto.com/ja/guides/glossary
## 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
[@yui.syui.ai](https://bsky.app/profile/did:plc:4hqjfn7m6n5hno3doamuhgef)
- https://pds.syu.is/at://did:plc:4hqjfn7m6n5hno3doamuhgef
```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://github.com/itaru2622/bluesky-selfhost-env/blob/master/patching/105-atproto-services-for-docker.diff)|
|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
View 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"
}

148
compose.yml Normal file
View File

@ -0,0 +1,148 @@
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
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
build:
context: ./repos/atproto/
dockerfile: services/pds/Dockerfile
restart: always
env_file:
- ./envs/pds
volumes:
- ./data/pds/:/data/
depends_on:
database:
condition: service_healthy
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
social-app:
ports:
- 8100:8100
build:
context: ./repos/social-app/
dockerfile: Dockerfile
restart: always
env_file:
- ./envs/social-app
command: "/usr/bin/bskyweb serve"
bsky:
ports:
- 2584:2584
build:
context: ./repos/atproto/
dockerfile: services/bsky/Dockerfile
restart: always
env_file:
- ./envs/bsky
user: root
volumes:
- ./data/bsky/:/data/
command: node --enable-source-maps api.js
depends_on:
database:
condition: service_healthy
redis:
condition: service_healthy
jetstream:
build:
context: ./repos/jetstream/
dockerfile: Dockerfile
ports:
- 6008:6008
volumes:
- ./data/jetstream:/data
restart: always
env_file:
- ./envs/jetstream
ozone-web:
build:
context: ./repos/ozone/
ports:
- 2586:3000
restart: always
volumes:
- ./data/ozone/:/data/
env_file:
- ./envs/ozone
depends_on:
database:
condition: service_healthy
ozone:
build:
context: ./repos/atproto/
dockerfile: services/ozone/Dockerfile
ports:
- 2585:3000
restart: always
command: node --enable-source-maps api.js
volumes:
- ./data/ozone/:/data/
env_file:
- ./envs/ozone
ozone-daemon:
build:
context: ./repos/atproto/
dockerfile: services/ozone/Dockerfile
restart: always
command: node --enable-source-maps daemon.js
volumes:
- ./data/ozone/:/data/
env_file:
- ./envs/ozone

View 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
View 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
View 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

5
envs/jetstream Normal file
View File

@ -0,0 +1,5 @@
JETSTREAM_WS_URL=wss://bgs.${host}/xrpc/com.atproto.sync.subscribeRepos
JETSTREAM_DATA_DIR=/data
JETSTREAM_LISTEN_ADDR=:6008
JETSTREAM_METRICS_LISTEN_ADDR=:6009
JETSTREAM_LIVENESS_TTL=96h

27
envs/ozone Normal file
View File

@ -0,0 +1,27 @@
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
OZONE_BLOB_DIVERT_ADMIN_PASSWORD
OZONE_VERIFIER_URL
OZONE_VERIFIER_DID
OZONE_VERIFIER_PASSWORD
OZONE_VERIFIER_ISSUERS_TO_INDEX
OZONE_VERIFIER_JETSTREAM_URL

21
envs/pds Normal file
View 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
View 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
View 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
View File

@ -0,0 +1 @@
ATP_APPVIEW_HOST=https://bsky.${host}

52
icons/Logotype.tsx Normal file
View 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
View 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

201
install.zsh Executable file
View File

@ -0,0 +1,201 @@
#!/bin/zsh
# ./install.zsh $HOST
repos_v='{}'
function at-repos-env() {
host=$1
if [ -z "$1" ];then
host=syu.is
fi
did=did:plc:6qyecktefllvenje24fcxnie
icon=https://git.syui.ai/ai/at/raw/branch/main/icons/Logotype.tsx
repos=(
https://github.com/did-method-plc/did-method-plc
https://github.com/bluesky-social/indigo
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/ozone
https://github.com/bluesky-social/jetstream
)
d=${0:a:h}
dh=${0:a:h:h}
name=${host%%.*}
domain=${host##*.}
}
function at-repos-json() {
f=~/.config/atproto/token.json
j="{ \"did\": \"did:plc:6qyecktefllvenje24fcxnie\", \"didDoc\": { \"service\": [ { \"serviceEndpoint\": \"https://syu.is\" } ] }, \"handle\": \"ai.syu.is\", \"accessJwt\": \"xxx\" }"
if [ ! -f "$f" ];then
mkdir -p ~/.config/atproto
echo $j >> $f
fi
echo $f
}
function at-repos-token() {
at-repos-json
if [ -z "$host" ] && [ -f $f ];then
host=`cat $f|jq -r ".didDoc.service.[].serviceEndpoint"`
handle=`cat $f|jq -r ".handle"`
did=`cat $f|jq -r ".did"`
token=`cat $f|jq -r ".token"`
host=${host##*/}
fi
name=${host%%.*}
domain=${host##*.}
}
function at-repos-clone() {
if [ ! -d $d/repos ];then
mkdir -p $d/repos
fi
cd $d/repos
for ((i=1; i<=${#repos}; i++)); do
repo=${repos[$i]}
echo $repo
if [ ! -d $d/repos/${repo##*/} ];then
git clone $repo
fi
done
if [ ! -f $d/repos/feed-generator/Dockerfile ] && [ -f $d/docker/feed/Dockerfile ];then
cp -rf $d/docker/feed/Dockerfile $d/repos/feed-generator/
fi
}
function at-repos-pull() {
cd $d/repos
for ((i=1; i<=${#repos}; i++)); do
repo=${repos[$i]}
echo $repo
if [ -d $d/repos/${repo##*/} ];then
cd $d/repos/${repo##*/}
if ! git pull;then
rm -rf $d/repos/${repo##*/}
at-repos-clone
fi
fi
rv=$(echo "$repos_v" | jq -r ".[\"${repo##*/}\"]")
if [ "$rv" != "null" ];then
cd $d/repos/${repo##*/}
git reset --hard $rv
cd ..
fi
done
cd $d
}
function at-repos-social-app-icon() {
curl -sL https://raw.githubusercontent.com/bluesky-social/social-app/main/src/view/icons/Logotype.tsx -o $d/repos/social-app/src/view/icons/Logotype.tsx
if [ -d $d/icons ];then
mkdir -p $d/icons
fi
cp -rf $d/repos/social-app/src/view/icons/Logotype.tsx $d/icons/
}
function at-repos-social-app-icon-origin() {
curl -sL $icon -o $d/icons/Logotype.tsx
}
function at-repos-social-app-write() {
did_admin=did:plc:6qyecktefllvenje24fcxnie
dt=$d/repos/social-app/src
cd $dt
grep -R syu.is .|cut -d : -f 1|sort -u|xargs sed -i "s/syu.is/${host}/g"
grep -R web.syu.is .|cut -d : -f 1|sort -u|xargs sed -i "s/web.syu.is/web.${host}/g"
f=$dt/lib/constants.ts
sed -i "s/public.api.web/bsky/g" $f
f=$dt/view/icons/Logotype.tsx
o=$d/icons/Logotype.tsx
cp -rf $o $f
f=$dt/view/com/util/UserAvatar.tsx
curl -sL https://raw.githubusercontent.com/bluesky-social/social-app/refs/heads/main/src/view/com/util/UserAvatar.tsx -o $f
sed -i "s#/img/avatar/plain/#https://cdn.web.syu.is/img/avatar/plain/#g" $f
sed -i "s#/img/avatar_thumbnail/plain/#https://bsky.${host}/img/avatar/plain/#g" $f
sed -i "s#source={{uri: avatar}}#source={{ uri: hackModifyThumbnailPath(avatar, 1 > 0), }}#g" $f
curl -sL https://raw.githubusercontent.com/bluesky-social/social-app/refs/heads/main/src/lib/strings/url-helpers.ts -o $dt/lib/strings/url-helpers.ts
sed -i "s#https://go.web.syu.is/redirect?u=\${encodeURIComponent(url)}#\${url}#g" $dt/lib/strings/url-helpers.ts
grep -R $did_admin .|cut -d : -f 1|sort -u|xargs sed -i "s/${did_admin}/${did}/g"
}
function at-repos-bsky-patch() {
f=$d/repos/atproto/services/bsky/api.js
curl -sL https://raw.githubusercontent.com/bluesky-social/atproto/refs/heads/main/services/bsky/api.js -o $f
d_=$d/repos/atproto
p_=$d/patching/105-atproto-services-for-docker.diff
cd ${d_}
if [ ! -f ${p_} ];then
# https://github.com/itaru2622/bluesky-selfhost-env/blob/master/patching/105-atproto-services-for-docker.diff
echo download patch: https://github.com/itaru2622/bluesky-selfhost-env/blob/master/patching/105-atproto-services-for-docker.diff
curl -sL https://raw.githubusercontent.com/itaru2622/bluesky-selfhost-env/refs/heads/master/patching/105-atproto-services-for-docker.diff -o ${p_}
else
echo local patch
fi
echo "applying patch: under ${f} for ${p_}"
pushd ${d_}
patch -p1 < ${p_}
popd
}
function at-repos-social-app-patch() {
f=$d/repos/social-app/Dockerfile
p_=$d/patching/social-app-dockerfile.diff
d_=$d/repos/social-app
cd ${d_}
curl -sLO https://raw.githubusercontent.com/bluesky-social/social-app/refs/heads/main/Dockerfile
echo "applying patch: under ${f} for ${p_}"
pushd ${d_}
patch -p1 < ${p_}
popd
}
function at-repos-ozone-patch() {
#DOMAIN=syu.is
cd $d/repos
d_=$d/repos/ozone
rm -rf ${d_}
p_=$d/patching/120-ozone-runtimeEnvVars.diff
git clone https://github.com/bluesky-social/ozone
cd ${d_}
pushd ${d_}
echo "applying patch: under ${d_} for ${p_}"
patch -p1 < ${p_}
popd
p_=$d/patching/122-ozone-enable-daemon.diff
echo "applying patch: under ${d_} for ${p_}"
pushd ${d_}
patch -p1 < ${p_}
popd
#cp -rf $d/repos/atproto/service/ozone/* $d/ozone/service/
}
function at-repos-docker() {
cd $d
docker compose build && docker compose up -d
}
at-repos-env
at-repos-clone
at-repos-pull
at-repos-social-app-icon
at-repos-social-app-icon-origin
at-repos-social-app-write
at-repos-bsky-patch
at-repos-social-app-patch
at-repos-ozone-patch
at-repos-docker
#echo "[y]docker compose build && up"
#read key
#case $key in
# [yY])
# at-repos-docker
# ;;
#esac

BIN
item/card/0.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 44 KiB

BIN
item/card/0.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

BIN
item/card/1.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 544 KiB

BIN
item/card/1.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 69 KiB

BIN
item/card/10.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 933 KiB

BIN
item/card/10.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 160 KiB

BIN
item/card/11.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 688 KiB

BIN
item/card/11.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 136 KiB

BIN
item/card/12.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 516 KiB

BIN
item/card/12.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 60 KiB

BIN
item/card/13.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 324 KiB

BIN
item/card/13.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 105 KiB

BIN
item/card/135.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 38 KiB

BIN
item/card/135.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

BIN
item/card/136.mp4 Normal file

Binary file not shown.

BIN
item/card/136.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 MiB

BIN
item/card/136.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 MiB

BIN
item/card/14.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 462 KiB

BIN
item/card/14.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 109 KiB

BIN
item/card/2.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 361 KiB

BIN
item/card/2.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 39 KiB

BIN
item/card/3.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 683 KiB

BIN
item/card/3.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 94 KiB

BIN
item/card/4.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 805 KiB

BIN
item/card/4.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 94 KiB

BIN
item/card/5.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 402 KiB

BIN
item/card/5.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 81 KiB

BIN
item/card/6.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 476 KiB

BIN
item/card/6.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 104 KiB

BIN
item/card/7.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 713 KiB

BIN
item/card/7.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 90 KiB

BIN
item/card/8.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 330 KiB

BIN
item/card/8.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 51 KiB

BIN
item/card/9.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 680 KiB

BIN
item/card/9.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 103 KiB

38
item/card/readme.md Normal file
View File

@ -0,0 +1,38 @@
# card
カードの仕様を決定します。
基本的に`3:4``1620x2160`で生成していましたが、`4:5``1080x1350`で生成することにしました。これはwindows動画編集が4:5にしか対応していないからです。`.mov`から動画編集で`.mp4`に変換し、それを`.webp`にします。
comfyuiの動画生成はカードの大きさではなかなか思うような動画が作れませんでした。やはり`1920x1080(1080p)`からの生成が最適でした。
- [16:9] 1920x1080 (1080p)
- [16:9] 1280×720 (720p)
- [16:9] 854x480 (480p)
- [4:3] 2160x1620
- [4:3] 1440x1080
- [4:3] 1024×768
- [4:3] 640×480
- [4:5] 2025x1620
- [4:5] 1536x2048
- [4:5] 1080x1350
```bash
$ magick convert -resize 480x -quality 100 4.png 4.webp
$ magick convert -quality 100 1.png 1.webp
$ magick convert -size 1620x2160 xc:white 1080p.png
$ magick convert -resize 1080x -quality 100 1.png 1.webp
```
```sh
$ ffmpeg -i 4.mp4 -vf "fps=15,setpts=PTS/4" -c:v libwebp -loop 1 -quality 100 4.webp
```
## list
<img src="https://git.syui.ai/ai/ai/raw/branch/main/card/1.webp" width="200px">
<img src="https://git.syui.ai/ai/ai/raw/branch/main/card/2.webp" width="200px">
<img src="https://git.syui.ai/ai/ai/raw/branch/main/card/3.webp" width="200px">
<img src="https://git.syui.ai/ai/ai/raw/branch/main/card/4.webp" width="200px">

153
lex/ai.syui.system.json Normal file
View File

@ -0,0 +1,153 @@
{
"lexicon": 1,
"id": "ai.syui.system",
"revision": 1,
"description": "Syui's system definitions for game logic and metaphysics",
"defs": {
"main": {
"type": "record",
"record": {
"key": "self",
"recordType": "document",
"schema": {
"type": "object",
"required": ["card", "system"],
"properties": {
"card": {
"type": "array",
"items": { "$ref": "#card" }
},
"system": {
"type": "array",
"items": { "$ref": "#system" }
},
"weapon": {
"type": "array",
"items": { "$ref": "#weapon" }
},
"ability": {
"type": "array",
"items": { "$ref": "#ability" }
},
"character": {
"type": "array",
"items": { "$ref": "#character" }
}
}
}
}
},
"card": {
"type": "object",
"required": ["id", "img", "name", "lang"],
"properties": {
"id": { "type": "integer" },
"img": { "type": "string" },
"name": { "type": "string" },
"lang": {
"type": "object",
"properties": {
"ja": {
"type": "object",
"properties": {
"name": { "type": "string" },
"text": { "type": "string" }
}
}
}
}
}
},
"system": {
"type": "object",
"required": ["id", "enum", "name", "lang"],
"properties": {
"id": { "type": "integer" },
"enum": {
"type": "array",
"items": { "type": "string" }
},
"name": { "type": "string" },
"alias": {
"type": "array",
"items": { "type": "string" }
},
"lang": {
"type": "object",
"properties": {
"ja": { "type": "string" }
}
},
"description": { "type": "string" }
}
},
"weapon": {
"type": "object",
"required": ["id", "img", "name", "lang"],
"properties": {
"id": { "type": "integer" },
"img": { "type": "string" },
"name": { "type": "string" },
"lang": {
"type": "object",
"properties": {
"ja": {
"type": "object",
"properties": {
"name": { "type": "string" },
"text": { "type": "string" }
}
}
}
}
}
},
"ability": {
"type": "object",
"required": ["id", "name", "color"],
"properties": {
"id": { "type": "integer" },
"name": { "type": "string" },
"lang": {
"type": "object",
"properties": {
"ja": { "type": "string" }
}
},
"color": { "type": "string" },
"group": {
"type": "array",
"items": { "type": "string" }
},
"duration": { "type": "integer" },
"multiplier": {
"type": "object",
"additionalProperties": { "type": "number" }
}
}
},
"character": {
"type": "object",
"required": ["id", "name", "lang"],
"properties": {
"id": { "type": "integer" },
"img": { "type": "string" },
"name": { "type": "string" },
"lang": {
"type": "object",
"properties": {
"ja": {
"type": "object",
"properties": {
"name": { "type": "string" },
"text": { "type": "string" },
"description": { "type": "string" }
}
}
}
},
"text": { "type": "string" }
}
}
}
}

View File

@ -0,0 +1,56 @@
{
"lexicon": 1,
"id": "ai.syui.card",
"defs": {
"main": {
"type": "record",
"description": "Record containing a cards box.",
"key": "tid",
"record": {
"type": "object",
"required": ["verify", "createdAt"],
"properties": {
"id":{
"type": "integer",
"minimum": 0,
"maximum": 14,
"default": 0
},
"cp":{
"type": "integer",
"minimum": 1,
"maximum": 5000,
"default": 1
},
"rank":{
"type": "integer",
"minimum": 0,
"maximum": 7,
"default": 0
},
"rare": {
"type": "string",
"enum": ["normal", "super", "ultra", "yui", "ai"],
"default": "normal"
},
"author": {
"type": "string",
"format": "uri",
"description": "https://verify...",
"default": "https://yui.syui.ai"
},
"verify": {
"type": "string",
"format": "at-uri",
"description": "at://verify..."
},
"createdAt": {
"type": "string",
"format": "datetime",
"description": "Client-declared timestamp when this post was originally created."
}
}
}
}
}
}

View File

@ -0,0 +1,72 @@
{
"lexicon": 1,
"id": "ai.syui.card.verify",
"defs": {
"main": {
"type": "record",
"description": "Record containing a card verify.",
"key": "tid",
"record": {
"type": "object",
"required": [
"handle",
"did",
"createdAt"
],
"properties": {
"id": {
"type": "integer",
"minimum": 0,
"maximum": 14,
"default": 0
},
"cp": {
"type": "integer",
"minimum": 1,
"maximum": 10000,
"default": 1
},
"rank": {
"type": "integer",
"minimum": 0,
"maximum": 7,
"default": 0
},
"rare": {
"type": "string",
"enum": [
"normal",
"super",
"ultra",
"yui",
"ai"
],
"default": "normal"
},
"handle": {
"type": "string",
"maxLength": 32,
"maxGraphemes": 32
},
"did": {
"type": "string"
},
"embed": {
"type": "union",
"refs": [
"app.bsky.embed.images",
"app.bsky.embed.external",
"app.bsky.embed.record",
"app.bsky.embed.recordWithMedia"
]
},
"createdAt": {
"type": "string",
"format": "datetime",
"description": "Client-declared timestamp when this post was originally created."
}
}
}
}
}
}

View File

@ -0,0 +1,27 @@
{
"lexicon": 1,
"id": "ai.syui.game",
"defs": {
"main": {
"type": "record",
"description": "Record containing a game.",
"key": "literal:self",
"record": {
"type": "object",
"required": ["account", "createdAt"],
"properties": {
"account": {
"type": "string",
"format": "at-uri",
"description": "at://verify..."
},
"createdAt": {
"type": "string",
"format": "datetime",
"description": "Client-declared timestamp when this post was originally created."
}
}
}
}
}
}

View File

@ -0,0 +1,80 @@
{
"lexicon": 1,
"id": "ai.syui.game.character",
"defs": {
"main": {
"type": "record",
"key": "string",
"description": "record containing a game character.",
"input": {
"encoding": "application/json",
"record": {
"type": "object",
"required": [
"createdAt"
],
"charactor": {
"type": "object",
"enum": [
"ai",
"manny",
"quinn",
"chinese",
"phoenix",
"kirin",
"leviathan",
"wyvern",
"cerberus",
"dragon",
"kitsune",
"pegasus"
],
"properties": {
"type": "object",
"properties": {
"season": {
"type": "integer",
"minimum": 0,
"maximum": 3,
"default": 1
},
"group": {
"type": "string",
"default": "fantasy",
"enum": [
"origin",
"fantasy",
"animal"
]
},
"gender": {
"type": "string",
"enum": [
"none",
"male",
"famale"
]
},
"fullname": {
"type": "string"
},
"nickname": {
"type": "string"
}
}
}
},
"createdAt": {
"type": "string",
"format": "datetime",
"description": "Client-declared timestamp when this post was originally created."
},
"updatedAt": {
"type": "string",
"format": "datetime"
}
}
}
}
}
}

View File

@ -0,0 +1,101 @@
{
"lexicon": 1,
"id": "ai.syui.game.user",
"defs": {
"main": {
"type": "record",
"key": "tid",
"description": "Record containing a game user.",
"input": {
"encoding": "application/json",
"record": {
"type": "object",
"required": ["did", "createdAt"],
"properties": {
"aiten":{
"type": "integer",
"default": 0
},
"did": {
"type": "string"
},
"login": {
"type": "bool"
},
"limit": {
"type": "bool"
},
"charactor": {
"type": "object",
"enum": ["ai","cerberus","chinesedragon","dragon","kirin","kitsune","leviathan","pegasus","phoenix"],
"properties": {
"type": "object",
"properties": {
"group":{
"type": "string"
},
"season":{
"type": "integer"
},
"lv":{
"type": "integer",
"minimum": 1,
"maximum": 7,
"default": 1
},
"exp":{
"type": "integer"
},
"rank":{
"type": "integer",
"minimum": 0,
"maximum": 7,
"default": 0
},
"mode":{
"type": "integer",
"minimum": 0,
"maximum": 7,
"default": 0
},
"hp":{
"type": "integer",
"maximum": 255,
"default": 0
},
"attach":{
"type": "integer",
"minimum": 1,
"maximum": 255,
"default": 1
},
"critical":{
"type": "integer",
"minimum": 0,
"maximum": 255,
"default": 0
},
"critical_d":{
"type": "integer",
"minimum": 0,
"maximum": 255,
"default": 0
}
}
}
},
"createdAt": {
"type": "string",
"format": "datetime",
"description": "Client-declared timestamp when this post was originally created."
},
"updatedAt": {
"type": "string",
"format": "datetime"
}
}
}
}
}
}
}

View 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" }
}
}
}
}
}

View 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."
}
}
}
}
}
}

View 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."
}
}
}
}
}
}

View 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": {}
}
}
}

View 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."
}
]
}
}
}

View 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" }
}
}
}
}

View 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" }]
}
}
}

View 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)"
}
}
}
}
}
}
}

View 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" }]
}
}
}

View 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"
}
}
}
}

View 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" }
}
}
}
}

View 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" }
}
}
}
}

View 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" }]
}
}
}

View 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" }
}
}
}
}

View 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
patching/.keep Normal file
View File

View File

@ -0,0 +1,85 @@
diff --git a/services/bsky/api.js b/services/bsky/api.js
index 56c769b9d..5d14c0057 100644
--- a/services/bsky/api.js
+++ b/services/bsky/api.js
@@ -44,19 +44,62 @@ const assert = require('node:assert')
const cluster = require('node:cluster')
const path = require('node:path')
-const { BskyAppView, ServerConfig } = require('@atproto/bsky')
+const bsky = require('@atproto/bsky') // import all bsky features
const { Secp256k1Keypair } = require('@atproto/crypto')
const main = async () => {
const env = getEnv()
- const config = ServerConfig.readEnv()
+ const config = bsky.ServerConfig.readEnv()
assert(env.serviceSigningKey, 'must set BSKY_SERVICE_SIGNING_KEY')
const signingKey = await Secp256k1Keypair.import(env.serviceSigningKey)
- const bsky = BskyAppView.create({ config, signingKey })
- await bsky.start()
+
+// starts: involve logics in packages/dev-env/src/bsky.ts >>>>>>>>>>>>>
+// Separate migration db in case migration changes some connection state that we need in the tests, e.g. "alter database ... set ..."
+ const migrationDb = new bsky.Database({
+ url: env.dbPostgresUrl,
+ schema: env.dbPostgresSchema,
+ })
+ if (env.migration) {
+ await migrationDb.migrateToOrThrow(env.migration)
+ } else {
+ await migrationDb.migrateToLatestOrThrow()
+ }
+ await migrationDb.close()
+
+ const db = new bsky.Database({
+ url: env.dbPostgresUrl,
+ schema: env.dbPostgresSchema,
+ poolSize: 10,
+ })
+
+ const dataplane = await bsky.DataPlaneServer.create(
+ db,
+ env.dataplanePort,
+ config.didPlcUrl
+ )
+
+ const bsync = await bsky.MockBsync.create(db, env.bsyncPort)
+
+// ends: involve logics in packages/dev-env/src/bsky.ts <<<<<<<<<<<<<
+
+ const server = bsky.BskyAppView.create({ config, signingKey })
+// starts: involve logics in packages/dev-env/src/bsky.ts >>>>>>>>>>>>>
+ const sub = new bsky.RepoSubscription({
+ service: env.repoProvider,
+ db,
+ idResolver: dataplane.idResolver,
+ background: new bsky.BackgroundQueue(db),
+ })
+// ends: involve logics in packages/dev-env/src/bsky.ts <<<<<<<<<<<<<
+ await server.start()
+ sub.start() // involve logics in packages/dev-env/src/bsky.ts
// Graceful shutdown (see also https://aws.amazon.com/blogs/containers/graceful-shutdowns-with-ecs/)
const shutdown = async () => {
- await bsky.destroy()
+ await server.destroy()
+ await bsync.destroy()
+ await dataplane.destroy()
+ await sub.destroy()
+ await db.close()
}
process.on('SIGTERM', shutdown)
process.on('disconnect', shutdown) // when clustering
@@ -64,6 +107,12 @@ const main = async () => {
const getEnv = () => ({
serviceSigningKey: process.env.BSKY_SERVICE_SIGNING_KEY || undefined,
+ dbPostgresUrl: process.env.BSKY_DB_POSTGRES_URL || undefined,
+ dbPostgresSchema: process.env.BSKY_DB_POSTGRES_SCHEMA || undefined,
+ dataplanePort : maybeParseInt(process.env.BSKY_DATAPLANE_PORT) || undefined,
+ bsyncPort : maybeParseInt(process.env.BSKY_BSYNC_PORT) || undefined,
+ migration: process.env.ENABLE_MIGRATIONS === 'true' || undefined,
+ repoProvider: process.env.BSKY_REPO_PROVIDER || undefined
})
const maybeParseInt = (str) => {

View File

@ -0,0 +1,119 @@
diff --git a/app/layout.tsx b/app/layout.tsx
index bfc3470..9350629 100644
--- a/app/layout.tsx
+++ b/app/layout.tsx
@@ -5,6 +5,7 @@ import 'yet-another-react-lightbox/styles.css'
import 'yet-another-react-lightbox/plugins/thumbnails.css'
import 'yet-another-react-lightbox/plugins/captions.css'
import { ToastContainer } from 'react-toastify'
+import { PublicEnvScript } from 'next-runtime-env';
import { Shell } from '@/shell/Shell'
import { CommandPaletteRoot } from '@/shell/CommandPalette/Root'
@@ -36,6 +37,7 @@ export default function RootLayout({
isDarkModeEnabled() ? 'dark' : ''
}`}
>
+ <head>
<title>Ozone</title>
<link
rel="icon"
@@ -43,6 +45,8 @@ export default function RootLayout({
sizes="any"
/>
<meta name="viewport" content="width=device-width, initial-scale=1" />
+ <PublicEnvScript />
+ </head>
<body className="h-full overflow-hidden">
<ToastContainer
position="bottom-right"
diff --git a/environment.d.ts b/environment.d.ts
index 7a47cc2..33ab29f 100644
--- a/environment.d.ts
+++ b/environment.d.ts
@@ -9,6 +9,8 @@ declare global {
NEXT_PUBLIC_OZONE_SERVICE_DID?: string // e.g. did:plc:xxx#atproto_labeler
NEXT_PUBLIC_OZONE_PUBLIC_URL?: string // e.g. https://ozone.example.com (falls back to window.location.origin)
NEXT_PUBLIC_SOCIAL_APP_URL?: string // e.g. https://bsky.app
+ NEXT_PUBLIC_SOCIAL_APP_DOMAIN?: string // e.g. bsky.app
+ NEXT_PUBLIC_HANDLE_RESOLVER_URL?: string // e.g. https://api.bsky.app
}
}
}
diff --git a/lib/constants.ts b/lib/constants.ts
index fe7c0e8..c286ac6 100644
--- a/lib/constants.ts
+++ b/lib/constants.ts
@@ -1,29 +1,32 @@
+import { env } from 'next-runtime-env';
+
export const OAUTH_SCOPE = 'atproto transition:generic'
export const OZONE_SERVICE_DID =
- process.env.NEXT_PUBLIC_OZONE_SERVICE_DID || undefined
+ env('NEXT_PUBLIC_OZONE_SERVICE_DID') || undefined
export const OZONE_PUBLIC_URL =
- process.env.NEXT_PUBLIC_OZONE_PUBLIC_URL || undefined
+ env('NEXT_PUBLIC_OZONE_PUBLIC_URL') || undefined
export const PLC_DIRECTORY_URL =
- process.env.NEXT_PUBLIC_PLC_DIRECTORY_URL ||
+ env('NEXT_PUBLIC_PLC_DIRECTORY_URL') ||
(process.env.NODE_ENV === 'development'
? 'http://localhost:2582'
: 'https://plc.directory')
-export const QUEUE_CONFIG = process.env.NEXT_PUBLIC_QUEUE_CONFIG || '{}'
+export const QUEUE_CONFIG = env('NEXT_PUBLIC_QUEUE_CONFIG') || '{}'
-export const QUEUE_SEED = process.env.NEXT_PUBLIC_QUEUE_SEED || ''
+export const QUEUE_SEED = env('NEXT_PUBLIC_QUEUE_SEED') || ''
+export const SOCIAL_APP_DOMAIN = env('NEXT_PUBLIC_SOCIAL_APP_DOMAIN') || 'bsky.app'
export const SOCIAL_APP_URL =
- process.env.NEXT_PUBLIC_SOCIAL_APP_URL ||
+ env('NEXT_PUBLIC_SOCIAL_APP_URL') ||
(process.env.NODE_ENV === 'development'
? 'http://localhost:2584'
- : 'https://bsky.app')
+ : `https://${SOCIAL_APP_DOMAIN}`)
export const HANDLE_RESOLVER_URL =
- process.env.NEXT_PUBLIC_HANDLE_RESOLVER_URL ||
+ env('NEXT_PUBLIC_HANDLE_RESOLVER_URL') ||
(process.env.NODE_ENV === 'development'
? 'http://localhost:2584'
: 'https://api.bsky.app')
diff --git a/lib/util.ts b/lib/util.ts
index 0aa4460..ecec7d1 100644
--- a/lib/util.ts
+++ b/lib/util.ts
@@ -1,5 +1,5 @@
import { CollectionId } from '@/reports/helpers/subject'
-import { SOCIAL_APP_URL } from './constants'
+import { SOCIAL_APP_URL, SOCIAL_APP_DOMAIN } from './constants'
import { AtUri } from '@atproto/api'
export function classNames(...classes: (string | undefined)[]) {
@@ -57,7 +57,7 @@ export function takesKeyboardEvt(el?: EventTarget | null) {
)
}
-const blueSkyUrlMatcher = new RegExp('(https?://)?.*bsky.app')
+const blueSkyUrlMatcher = new RegExp('(https?://)?.*'+ `${SOCIAL_APP_DOMAIN}`)
export const isBlueSkyAppUrl = (url: string) => blueSkyUrlMatcher.test(url)
diff --git a/package.json b/package.json
index 8919841..750dce9 100644
--- a/package.json
+++ b/package.json
@@ -37,6 +37,7 @@
"kbar": "^0.1.0-beta.45",
"lande": "^1.0.10",
"next": "15.2.4",
+ "next-runtime-env": "^3.2.1",
"react": "19.1.0",
"react-dom": "19.1.0",
"react-dropzone": "^14.3.5",

View File

@ -0,0 +1,82 @@
diff --git a/service/index.js b/service/index.js
index 943c281..7721cd9 100644
--- a/service/index.js
+++ b/service/index.js
@@ -1,5 +1,7 @@
const next = require('next')
-const {
+const ozone = require('@atproto/ozone')
+/*
+{
readEnv,
httpLogger,
envToCfg,
@@ -7,6 +9,7 @@ const {
OzoneService,
Database,
} = require('@atproto/ozone')
+*/
const pkg = require('@atproto/ozone/package.json')
async function main() {
@@ -16,37 +19,48 @@ async function main() {
const frontendHandler = frontend.getRequestHandler()
await frontend.prepare()
// backend
- const env = readEnv()
+ const env = ozone.readEnv()
env.version ??= pkg.version
- const config = envToCfg(env)
- const secrets = envToSecrets(env)
+ const config = ozone.envToCfg(env)
+ const secrets = ozone.envToSecrets(env)
const migrate = process.env.OZONE_DB_MIGRATE === '1'
if (migrate) {
- const db = new Database({
+ const db = new ozone.Database({
url: config.db.postgresUrl,
schema: config.db.postgresSchema,
})
await db.migrateToLatestOrThrow()
await db.close()
}
- const ozone = await OzoneService.create(config, secrets)
+ const server = await ozone.OzoneService.create(config, secrets)
// setup handlers
- ozone.app.get('/.well-known/ozone-metadata.json', (_req, res) => {
+ server.app.get('/.well-known/ozone-metadata.json', (_req, res) => {
return res.json({
- did: ozone.ctx.cfg.service.did,
- url: ozone.ctx.cfg.service.publicUrl,
- publicKey: ozone.ctx.signingKey.did(),
+ did: server.ctx.cfg.service.did,
+ url: server.ctx.cfg.service.publicUrl,
+ publicKey: server.ctx.signingKey.did(),
})
})
// Note: We must use `use()` here. This should be the last middleware.
- ozone.app.use((req, res) => {
+ server.app.use((req, res) => {
void frontendHandler(req, res, undefined)
})
// run
- const httpServer = await ozone.start()
+ const httpServer = await server.start()
+ // starts: involve ops from atproto/packages/dev-env/src/ozone.ts >>>
+ ozone.httpLogger.info('starts ozone daemon')
+ const daemon = await ozone.OzoneDaemon.create(config, secrets)
+ await daemon.start()
+ //if (process.env.OZONE_ENABLE_EVENT_REVERSER != 'true') // atproto/services/ozone/daemon.js doesn't stop eventReverser
+ //{
+ // ozone.httpLogger.info('disable ozone daemon eventReverser')
+ // await daemon.ctx.eventReverser.destroy()
+ //}
+ // ends: involve ops from atproto/packages/dev-env/src/ozone.ts <<<
+
/** @type {import('net').AddressInfo} */
const addr = httpServer.address()
- httpLogger.info(`Ozone is running at http://localhost:${addr.port}`)
+ ozone.httpLogger.info(`Ozone is running at http://localhost:${addr.port}`)
}
main().catch(console.error)

View File

@ -0,0 +1,10 @@
--- a/Dockerfile
+++ b/Dockerfile
@@ -37,6 +37,7 @@ RUN \. "$NVM_DIR/nvm.sh" && \
echo "EXPO_PUBLIC_BUNDLE_IDENTIFIER=$EXPO_PUBLIC_BUNDLE_IDENTIFIER" >> .env && \
echo "EXPO_PUBLIC_BUNDLE_DATE=$(date -u +"%y%m%d%H")" >> .env && \
npm install --global yarn && \
+ yarn config set network-timeout 600000 && \
yarn && \
yarn intl:build && \
EXPO_PUBLIC_BUNDLE_IDENTIFIER=$EXPO_PUBLIC_BUNDLE_IDENTIFIER EXPO_PUBLIC_BUNDLE_DATE=$() yarn build-web

0
repos_extra/.keep Normal file
View File

View 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&apos;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>

View 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&apos;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}

View 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" }
}
}
}
}
}

View 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."
}
}
}
}
}
}

View 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."
}
}
}
}
}
}

View 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

View 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"]

View File

@ -0,0 +1,9 @@
services:
drainpipe:
build:
dockerfile: ./packages-rs/drainpipe/Dockerfile
context: ../../
env_file:
- ./.env.local
volumes:
- ./drainpipedata:/drainpipedata

View 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"]

View File

@ -0,0 +1,8 @@
services:
atbrowser:
build:
context: .
ports:
- "3000:3000"
environment:
- WATCHPACK_POLLING=true

View 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

View 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"]

View File

@ -0,0 +1,4 @@
<Button asChild>
<Link href="/post/game">Game</Link>
</Button>

View File

@ -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" };
}
}

View File

@ -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>
);
}

View File

@ -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>
);
}

View File

@ -0,0 +1,8 @@
services:
frontpage:
build:
context: .
ports:
- "3000:3000"
environment:
- WATCHPACK_POLLING=true

View File

@ -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);
}

View File

@ -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());
}

Some files were not shown because too many files have changed in this diff Show More