Compare commits
	
		
			66 Commits
		
	
	
		
			c0e4dc63ea
			...
			test-oauth
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 4edde5293a | |||
| f0fdf678c8 | |||
| 820e47f634 | |||
| 4dac4a83e0 | |||
| fccf75949c | |||
| 6600a9e0cf | |||
| 0d79af5aa5 | |||
| db04af76ab | |||
| 5f0b09b555 | |||
| 8fa9e474d1 | |||
| 5339dd28b0 | |||
| 1e83b50e3f | |||
| 889ce8baa1 | |||
| 286b46c6e6 | |||
| b780d27ace | |||
| 831fcb7865 | |||
| 3f8bbff7c2 | |||
| 5cb73a9ed3 | |||
| 6ce8d44c4b | |||
| 167cfb35f7 | |||
| c8377ceabf | |||
| e917c563f2 | |||
| a76933c23b | |||
| 8d960b7a40 | |||
| d3967c782f | |||
| 63b6fd5142 | |||
| 27935324c7 | |||
| 594d7e7aef | |||
| be86c11e74 | |||
| 619675b551 | |||
| d4d98e2e91 | |||
| 8dac463345 | |||
| 095f6ec386 | |||
| c12d42882c | |||
| 67b241f1e8 | |||
| 4206b2195d | |||
| b3c1b01e9e | |||
| ffa4fa0846 | |||
| 0e75d4c0e6 | |||
| b7f62e729a | |||
| 3b2c53fc97 | |||
| 13f1785081 | |||
| bb6d51a602 | |||
| a4114c5be3 | |||
| 5c13dc0a1c | |||
| cef0675a88 | |||
| fd223290df | |||
| 5f4382911b | |||
| 95cee69482 | |||
| 33c166fa0c | |||
| 36863e4d9f | |||
| fb0e5107cf | |||
| 962017f922 | |||
| 5ce03098bd | |||
| acce1d5af3 | |||
| bf0b72a52d | |||
| 6e6c6e2f53 | |||
| eb5aa0a2be | |||
| ad45b151b1 | |||
| 4775fa7034 | |||
| d396dbd052 | |||
| ec3e3d1f89 | |||
| b2fa06d5fa | |||
| bebd6a61eb | |||
| 4fe0582c6b | |||
| 637028c264 | 
| @@ -10,7 +10,49 @@ | |||||||
|       "Bash(/Users/syui/ai/log/target/debug/ailog build)", |       "Bash(/Users/syui/ai/log/target/debug/ailog build)", | ||||||
|       "Bash(ls:*)", |       "Bash(ls:*)", | ||||||
|       "Bash(curl:*)", |       "Bash(curl:*)", | ||||||
|       "Bash(pkill:*)" |       "Bash(pkill:*)", | ||||||
|  |       "WebFetch(domain:docs.anthropic.com)", | ||||||
|  |       "WebFetch(domain:github.com)", | ||||||
|  |       "Bash(rm:*)", | ||||||
|  |       "Bash(mv:*)", | ||||||
|  |       "Bash(cp:*)", | ||||||
|  |       "Bash(timeout:*)", | ||||||
|  |       "Bash(grep:*)", | ||||||
|  |       "Bash(./target/debug/ailog:*)", | ||||||
|  |       "Bash(cat:*)", | ||||||
|  |       "Bash(npm install)", | ||||||
|  |       "Bash(npm run build:*)", | ||||||
|  |       "Bash(chmod:*)", | ||||||
|  |       "Bash(./scripts/tunnel.sh:*)", | ||||||
|  |       "Bash(PRODUCTION=true cargo run -- build)", | ||||||
|  |       "Bash(cloudflared tunnel:*)", | ||||||
|  |       "Bash(npm install:*)", | ||||||
|  |       "Bash(./scripts/build-oauth-partial.zsh:*)", | ||||||
|  |       "Bash(./scripts/quick-oauth-update.zsh:*)", | ||||||
|  |       "Bash(../target/debug/ailog serve)", | ||||||
|  |       "Bash(./scripts/test-oauth.sh:*)", | ||||||
|  |       "Bash(./run.zsh:*)", | ||||||
|  |       "Bash(npm run dev:*)", | ||||||
|  |       "Bash(./target/release/ailog:*)", | ||||||
|  |       "Bash(rg:*)", | ||||||
|  |       "Bash(../target/release/ailog build)", | ||||||
|  |       "Bash(zsh run.zsh:*)", | ||||||
|  |       "Bash(hugo:*)", | ||||||
|  |       "WebFetch(domain:docs.bsky.app)", | ||||||
|  |       "WebFetch(domain:syui.ai)", | ||||||
|  |       "Bash(rustup target list:*)", | ||||||
|  |       "Bash(rustup target:*)", | ||||||
|  |       "Bash(git add:*)", | ||||||
|  |       "Bash(git commit:*)", | ||||||
|  |       "Bash(git push:*)", | ||||||
|  |       "Bash(git tag:*)", | ||||||
|  |       "Bash(../bin/ailog:*)", | ||||||
|  |       "Bash(../target/release/ailog oauth build:*)", | ||||||
|  |       "Bash(ailog:*)", | ||||||
|  |       "WebFetch(domain:plc.directory)", | ||||||
|  |       "WebFetch(domain:atproto.com)", | ||||||
|  |       "WebFetch(domain:syu.is)", | ||||||
|  |       "Bash(sed:*)" | ||||||
|     ], |     ], | ||||||
|     "deny": [] |     "deny": [] | ||||||
|   } |   } | ||||||
|   | |||||||
							
								
								
									
										53
									
								
								.gitea/workflows/deploy.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								.gitea/workflows/deploy.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | name: Deploy to Cloudflare Pages | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [main] | ||||||
|  |   pull_request: | ||||||
|  |     branches: [main] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build-and-deploy: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |      | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |        | ||||||
|  |     - name: Setup Rust | ||||||
|  |       uses: actions-rs/toolchain@v1 | ||||||
|  |       with: | ||||||
|  |         toolchain: stable | ||||||
|  |         override: true | ||||||
|  |          | ||||||
|  |     - name: Setup Node.js | ||||||
|  |       uses: actions/setup-node@v4 | ||||||
|  |       with: | ||||||
|  |         node-version: '20' | ||||||
|  |          | ||||||
|  |     - name: Build ailog | ||||||
|  |       run: | | ||||||
|  |         cargo build --release | ||||||
|  |          | ||||||
|  |     - name: Build OAuth app | ||||||
|  |       run: | | ||||||
|  |         cd oauth | ||||||
|  |         npm install | ||||||
|  |         npm run build | ||||||
|  |          | ||||||
|  |     - name: Copy OAuth assets | ||||||
|  |       run: | | ||||||
|  |         cp -r oauth/dist/* my-blog/static/ | ||||||
|  |          | ||||||
|  |     - name: Generate site with ailog | ||||||
|  |       run: | | ||||||
|  |         ./target/release/ailog generate --input content --output my-blog/public | ||||||
|  |          | ||||||
|  |     - name: Deploy to Cloudflare Pages | ||||||
|  |       uses: cloudflare/pages-action@v1 | ||||||
|  |       with: | ||||||
|  |         apiToken: ${{ secrets.CLOUDFLARE_API_TOKEN }} | ||||||
|  |         accountId: ${{ secrets.CLOUDFLARE_ACCOUNT_ID }} | ||||||
|  |         projectName: syui-ai | ||||||
|  |         directory: my-blog/public | ||||||
|  |         gitHubToken: ${{ secrets.GITHUB_TOKEN }} | ||||||
							
								
								
									
										28
									
								
								.gitea/workflows/example-usage.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								.gitea/workflows/example-usage.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | name: Example ailog usage | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   workflow_dispatch:  # Manual trigger for testing | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build-with-ailog-action: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |      | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |        | ||||||
|  |     - name: Build with ailog action | ||||||
|  |       uses: ai/log@v1  # This will reference this repository | ||||||
|  |       with: | ||||||
|  |         content-dir: 'content' | ||||||
|  |         output-dir: 'public' | ||||||
|  |         ai-integration: true | ||||||
|  |         atproto-integration: true | ||||||
|  |          | ||||||
|  |     - name: Deploy to Cloudflare Pages | ||||||
|  |       uses: cloudflare/pages-action@v1 | ||||||
|  |       with: | ||||||
|  |         apiToken: ${{ secrets.CLOUDFLARE_API_TOKEN }} | ||||||
|  |         accountId: ${{ secrets.CLOUDFLARE_ACCOUNT_ID }} | ||||||
|  |         projectName: my-blog | ||||||
|  |         directory: public | ||||||
							
								
								
									
										111
									
								
								.github/workflows/cloudflare-pages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								.github/workflows/cloudflare-pages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,111 @@ | |||||||
|  | name: Deploy to Cloudflare Pages | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - main | ||||||
|  |   workflow_dispatch: | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   deploy: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: read | ||||||
|  |       deployments: write | ||||||
|  |      | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|  |       - name: Setup Node.js | ||||||
|  |         uses: actions/setup-node@v4 | ||||||
|  |         with: | ||||||
|  |           node-version: '21' | ||||||
|  |  | ||||||
|  |       - name: Install dependencies | ||||||
|  |         run: | | ||||||
|  |           cd oauth | ||||||
|  |           npm install | ||||||
|  |  | ||||||
|  |       - name: Build OAuth app | ||||||
|  |         run: | | ||||||
|  |           cd oauth | ||||||
|  |           npm run build | ||||||
|  |            | ||||||
|  |       - name: Copy OAuth build to static | ||||||
|  |         run: | | ||||||
|  |           # Remove old assets (following run.zsh pattern) | ||||||
|  |           rm -rf my-blog/static/assets | ||||||
|  |           # Copy all dist files to static | ||||||
|  |           cp -rf oauth/dist/* my-blog/static/ | ||||||
|  |           # Copy index.html to oauth-assets.html template | ||||||
|  |           cp oauth/dist/index.html my-blog/templates/oauth-assets.html | ||||||
|  |            | ||||||
|  |       - name: Cache ailog binary | ||||||
|  |         uses: actions/cache@v4 | ||||||
|  |         with: | ||||||
|  |           path: ./bin | ||||||
|  |           key: ailog-bin-${{ runner.os }} | ||||||
|  |           restore-keys: | | ||||||
|  |             ailog-bin-${{ runner.os }} | ||||||
|  |              | ||||||
|  |       - name: Setup ailog binary | ||||||
|  |         run: | | ||||||
|  |           # Get expected version from Cargo.toml | ||||||
|  |           EXPECTED_VERSION=$(grep '^version' Cargo.toml | cut -d'"' -f2) | ||||||
|  |           echo "Expected version from Cargo.toml: $EXPECTED_VERSION" | ||||||
|  |            | ||||||
|  |           # Check current binary version if exists | ||||||
|  |           if [ -f "./bin/ailog" ]; then | ||||||
|  |             CURRENT_VERSION=$(./bin/ailog --version 2>/dev/null || echo "unknown") | ||||||
|  |             echo "Current binary version: $CURRENT_VERSION" | ||||||
|  |           else | ||||||
|  |             CURRENT_VERSION="none" | ||||||
|  |             echo "No binary found" | ||||||
|  |           fi | ||||||
|  |            | ||||||
|  |           # Check OS | ||||||
|  |           OS="${{ runner.os }}" | ||||||
|  |           echo "Runner OS: $OS" | ||||||
|  |            | ||||||
|  |           # Use pre-packaged binary if version matches or extract from tar.gz | ||||||
|  |           if [ "$CURRENT_VERSION" = "$EXPECTED_VERSION" ]; then | ||||||
|  |             echo "Binary is up to date" | ||||||
|  |             chmod +x ./bin/ailog | ||||||
|  |           elif [ "$OS" = "Linux" ] && [ -f "./bin/ailog-linux-x86_64.tar.gz" ]; then | ||||||
|  |             echo "Extracting ailog from pre-packaged tar.gz..." | ||||||
|  |             cd bin | ||||||
|  |             tar -xzf ailog-linux-x86_64.tar.gz | ||||||
|  |             chmod +x ailog | ||||||
|  |             cd .. | ||||||
|  |              | ||||||
|  |             # Verify extracted version | ||||||
|  |             EXTRACTED_VERSION=$(./bin/ailog --version 2>/dev/null || echo "unknown") | ||||||
|  |             echo "Extracted binary version: $EXTRACTED_VERSION" | ||||||
|  |              | ||||||
|  |             if [ "$EXTRACTED_VERSION" != "$EXPECTED_VERSION" ]; then | ||||||
|  |               echo "Warning: Binary version mismatch. Expected $EXPECTED_VERSION but got $EXTRACTED_VERSION" | ||||||
|  |             fi | ||||||
|  |           else | ||||||
|  |             echo "Error: No suitable binary found for OS: $OS" | ||||||
|  |             exit 1 | ||||||
|  |           fi | ||||||
|  |  | ||||||
|  |       - name: Build site with ailog | ||||||
|  |         run: | | ||||||
|  |           cd my-blog | ||||||
|  |           ../bin/ailog build | ||||||
|  |  | ||||||
|  |       - name: List public directory | ||||||
|  |         run: | | ||||||
|  |           ls -la my-blog/public/ | ||||||
|  |            | ||||||
|  |       - name: Deploy to Cloudflare Pages | ||||||
|  |         uses: cloudflare/pages-action@v1 | ||||||
|  |         with: | ||||||
|  |           apiToken: ${{ secrets.CLOUDFLARE_API_TOKEN }} | ||||||
|  |           accountId: ${{ secrets.CLOUDFLARE_ACCOUNT_ID }} | ||||||
|  |           projectName: ${{ secrets.CLOUDFLARE_PROJECT_NAME }} | ||||||
|  |           directory: my-blog/public | ||||||
|  |           gitHubToken: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |           wranglerVersion: '3' | ||||||
							
								
								
									
										92
									
								
								.github/workflows/disabled/gh-pages-fast.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								.github/workflows/disabled/gh-pages-fast.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | |||||||
|  | name: github pages (fast) | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |     - main | ||||||
|  |     paths-ignore: | ||||||
|  |       - 'src/**' | ||||||
|  |       - 'Cargo.toml' | ||||||
|  |       - 'Cargo.lock' | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build-deploy: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: write | ||||||
|  |       pages: write | ||||||
|  |       id-token: write | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |      | ||||||
|  |     - name: Cache ailog binary | ||||||
|  |       uses: actions/cache@v4 | ||||||
|  |       with: | ||||||
|  |         path: ./bin | ||||||
|  |         key: ailog-bin-${{ runner.os }} | ||||||
|  |         restore-keys: | | ||||||
|  |           ailog-bin-${{ runner.os }} | ||||||
|  |      | ||||||
|  |     - name: Setup ailog binary | ||||||
|  |       run: | | ||||||
|  |         # Get expected version from Cargo.toml | ||||||
|  |         EXPECTED_VERSION=$(grep '^version' Cargo.toml | cut -d'"' -f2) | ||||||
|  |         echo "Expected version from Cargo.toml: $EXPECTED_VERSION" | ||||||
|  |          | ||||||
|  |         # Check current binary version if exists | ||||||
|  |         if [ -f "./bin/ailog" ]; then | ||||||
|  |           CURRENT_VERSION=$(./bin/ailog --version 2>/dev/null || echo "unknown") | ||||||
|  |           echo "Current binary version: $CURRENT_VERSION" | ||||||
|  |         else | ||||||
|  |           CURRENT_VERSION="none" | ||||||
|  |           echo "No binary found" | ||||||
|  |         fi | ||||||
|  |          | ||||||
|  |         # Check OS | ||||||
|  |         OS="${{ runner.os }}" | ||||||
|  |         echo "Runner OS: $OS" | ||||||
|  |          | ||||||
|  |         # Use pre-packaged binary if version matches or extract from tar.gz | ||||||
|  |         if [ "$CURRENT_VERSION" = "$EXPECTED_VERSION" ]; then | ||||||
|  |           echo "Binary is up to date" | ||||||
|  |           chmod +x ./bin/ailog | ||||||
|  |         elif [ "$OS" = "Linux" ] && [ -f "./bin/ailog-linux-x86_64.tar.gz" ]; then | ||||||
|  |           echo "Extracting ailog from pre-packaged tar.gz..." | ||||||
|  |           cd bin | ||||||
|  |           tar -xzf ailog-linux-x86_64.tar.gz | ||||||
|  |           chmod +x ailog | ||||||
|  |           cd .. | ||||||
|  |            | ||||||
|  |           # Verify extracted version | ||||||
|  |           EXTRACTED_VERSION=$(./bin/ailog --version 2>/dev/null || echo "unknown") | ||||||
|  |           echo "Extracted binary version: $EXTRACTED_VERSION" | ||||||
|  |            | ||||||
|  |           if [ "$EXTRACTED_VERSION" != "$EXPECTED_VERSION" ]; then | ||||||
|  |             echo "Warning: Binary version mismatch. Expected $EXPECTED_VERSION but got $EXTRACTED_VERSION" | ||||||
|  |           fi | ||||||
|  |         else | ||||||
|  |           echo "Error: No suitable binary found for OS: $OS" | ||||||
|  |           exit 1 | ||||||
|  |         fi | ||||||
|  |        | ||||||
|  |     - name: Setup Hugo | ||||||
|  |       uses: peaceiris/actions-hugo@v3 | ||||||
|  |       with: | ||||||
|  |         hugo-version: "0.139.2" | ||||||
|  |         extended: true | ||||||
|  |          | ||||||
|  |     - name: Build with ailog | ||||||
|  |       env:  | ||||||
|  |         TZ: "Asia/Tokyo" | ||||||
|  |       run: | | ||||||
|  |           # Use pre-built ailog binary instead of cargo build | ||||||
|  |           cd my-blog | ||||||
|  |           ../bin/ailog build | ||||||
|  |           touch ./public/.nojekyll | ||||||
|  |            | ||||||
|  |     - name: Deploy | ||||||
|  |       uses: peaceiris/actions-gh-pages@v3 | ||||||
|  |       with: | ||||||
|  |         github_token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |         publish_dir: ./my-blog/public | ||||||
|  |         publish_branch: gh-pages | ||||||
							
								
								
									
										169
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										169
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,169 @@ | |||||||
|  | name: Release | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     tags: | ||||||
|  |       - 'v*' | ||||||
|  |   workflow_dispatch: | ||||||
|  |     inputs: | ||||||
|  |       tag: | ||||||
|  |         description: 'Release tag (e.g., v1.0.0)' | ||||||
|  |         required: true | ||||||
|  |         default: 'v0.1.0' | ||||||
|  |  | ||||||
|  | permissions: | ||||||
|  |   contents: write | ||||||
|  |   actions: read | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   CARGO_TERM_COLOR: always | ||||||
|  |   OPENSSL_STATIC: true | ||||||
|  |   OPENSSL_VENDOR: true | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     name: Build ${{ matrix.target }} | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     timeout-minutes: 60 | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         include: | ||||||
|  |           - target: x86_64-unknown-linux-gnu | ||||||
|  |             os: ubuntu-latest | ||||||
|  |             artifact_name: ailog | ||||||
|  |             asset_name: ailog-linux-x86_64 | ||||||
|  |           - target: aarch64-unknown-linux-gnu | ||||||
|  |             os: ubuntu-latest | ||||||
|  |             artifact_name: ailog | ||||||
|  |             asset_name: ailog-linux-aarch64 | ||||||
|  |           - target: x86_64-apple-darwin | ||||||
|  |             os: macos-latest | ||||||
|  |             artifact_name: ailog | ||||||
|  |             asset_name: ailog-macos-x86_64 | ||||||
|  |           - target: aarch64-apple-darwin | ||||||
|  |             os: macos-latest | ||||||
|  |             artifact_name: ailog | ||||||
|  |             asset_name: ailog-macos-aarch64 | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|  |     - name: Setup Rust | ||||||
|  |       uses: dtolnay/rust-toolchain@stable | ||||||
|  |       with: | ||||||
|  |         targets: ${{ matrix.target }} | ||||||
|  |  | ||||||
|  |     - name: Install cross-compilation tools (Linux) | ||||||
|  |       if: matrix.os == 'ubuntu-latest' && matrix.target == 'aarch64-unknown-linux-gnu' | ||||||
|  |       run: | | ||||||
|  |         sudo apt-get update | ||||||
|  |         sudo apt-get install -y gcc-aarch64-linux-gnu binutils-aarch64-linux-gnu | ||||||
|  |  | ||||||
|  |     - name: Configure cross-compilation (Linux ARM64) | ||||||
|  |       if: matrix.target == 'aarch64-unknown-linux-gnu' | ||||||
|  |       run: | | ||||||
|  |         echo '[target.aarch64-unknown-linux-gnu]' >> ~/.cargo/config.toml | ||||||
|  |         echo 'linker = "aarch64-linux-gnu-gcc"' >> ~/.cargo/config.toml | ||||||
|  |  | ||||||
|  |     - name: Cache cargo registry | ||||||
|  |       uses: actions/cache@v4 | ||||||
|  |       with: | ||||||
|  |         path: | | ||||||
|  |           ~/.cargo/registry | ||||||
|  |           ~/.cargo/git | ||||||
|  |         key: ${{ runner.os }}-${{ matrix.target }}-cargo-${{ hashFiles('**/Cargo.lock') }} | ||||||
|  |  | ||||||
|  |     - name: Cache target directory | ||||||
|  |       uses: actions/cache@v4 | ||||||
|  |       with: | ||||||
|  |         path: target | ||||||
|  |         key: ${{ runner.os }}-${{ matrix.target }}-target-${{ hashFiles('**/Cargo.lock') }} | ||||||
|  |  | ||||||
|  |     - name: Build | ||||||
|  |       run: cargo build --release --target ${{ matrix.target }} | ||||||
|  |  | ||||||
|  |     - name: Prepare binary | ||||||
|  |       shell: bash | ||||||
|  |       run: | | ||||||
|  |         cd target/${{ matrix.target }}/release | ||||||
|  |          | ||||||
|  |         # Use appropriate strip command for cross-compilation | ||||||
|  |         if [[ "${{ matrix.target }}" == "aarch64-unknown-linux-gnu" ]]; then | ||||||
|  |           aarch64-linux-gnu-strip ${{ matrix.artifact_name }} || echo "Strip failed, continuing..." | ||||||
|  |         elif [[ "${{ matrix.os }}" == "windows-latest" ]]; then | ||||||
|  |           strip ${{ matrix.artifact_name }} || echo "Strip failed, continuing..." | ||||||
|  |         else | ||||||
|  |           strip ${{ matrix.artifact_name }} || echo "Strip failed, continuing..." | ||||||
|  |         fi | ||||||
|  |          | ||||||
|  |         # Create archive | ||||||
|  |         if [[ "${{ matrix.target }}" == *"windows"* ]]; then | ||||||
|  |           7z a ../../../${{ matrix.asset_name }}.zip ${{ matrix.artifact_name }} | ||||||
|  |         else | ||||||
|  |           tar czvf ../../../${{ matrix.asset_name }}.tar.gz ${{ matrix.artifact_name }} | ||||||
|  |         fi | ||||||
|  |  | ||||||
|  |     - name: Upload binary | ||||||
|  |       uses: actions/upload-artifact@v4 | ||||||
|  |       with: | ||||||
|  |         name: ${{ matrix.asset_name }} | ||||||
|  |         path: ${{ matrix.asset_name }}.tar.gz | ||||||
|  |  | ||||||
|  |   release: | ||||||
|  |     name: Create Release | ||||||
|  |     needs: build | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: write | ||||||
|  |       actions: read | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|  |     - name: Download all artifacts | ||||||
|  |       uses: actions/download-artifact@v4 | ||||||
|  |       with: | ||||||
|  |         path: artifacts | ||||||
|  |  | ||||||
|  |     - name: Generate release notes | ||||||
|  |       run: | | ||||||
|  |         echo "## What's Changed" > release_notes.md | ||||||
|  |         echo "" >> release_notes.md | ||||||
|  |         echo "### Features" >> release_notes.md | ||||||
|  |         echo "- AI-powered static blog generator" >> release_notes.md | ||||||
|  |         echo "- AtProto OAuth integration" >> release_notes.md | ||||||
|  |         echo "- Automatic translation support" >> release_notes.md | ||||||
|  |         echo "- AI comment system" >> release_notes.md | ||||||
|  |         echo "" >> release_notes.md | ||||||
|  |         echo "### Platforms" >> release_notes.md | ||||||
|  |         echo "- Linux (x86_64, aarch64)" >> release_notes.md | ||||||
|  |         echo "- macOS (Intel, Apple Silicon)" >> release_notes.md | ||||||
|  |         echo "" >> release_notes.md | ||||||
|  |         echo "### Installation" >> release_notes.md | ||||||
|  |         echo "\`\`\`bash" >> release_notes.md | ||||||
|  |         echo "# Linux/macOS" >> release_notes.md | ||||||
|  |         echo "tar -xzf ailog-linux-x86_64.tar.gz" >> release_notes.md | ||||||
|  |         echo "chmod +x ailog" >> release_notes.md | ||||||
|  |         echo "sudo mv ailog /usr/local/bin/" >> release_notes.md | ||||||
|  |         echo "" >> release_notes.md | ||||||
|  |         echo "\`\`\`" >> release_notes.md | ||||||
|  |  | ||||||
|  |     - name: Get tag name | ||||||
|  |       id: tag_name | ||||||
|  |       run: | | ||||||
|  |         if [[ "${{ github.event_name }}" == "workflow_dispatch" ]]; then | ||||||
|  |           echo "tag=${{ github.event.inputs.tag }}" >> $GITHUB_OUTPUT | ||||||
|  |         else | ||||||
|  |           echo "tag=${GITHUB_REF#refs/tags/}" >> $GITHUB_OUTPUT | ||||||
|  |         fi | ||||||
|  |  | ||||||
|  |     - name: Create Release | ||||||
|  |       uses: softprops/action-gh-release@v1 | ||||||
|  |       with: | ||||||
|  |         tag_name: ${{ steps.tag_name.outputs.tag }} | ||||||
|  |         name: ailog ${{ steps.tag_name.outputs.tag }} | ||||||
|  |         body_path: release_notes.md | ||||||
|  |         draft: false | ||||||
|  |         prerelease: ${{ contains(steps.tag_name.outputs.tag, 'alpha') || contains(steps.tag_name.outputs.tag, 'beta') || contains(steps.tag_name.outputs.tag, 'rc') }} | ||||||
|  |         files: artifacts/*/ailog-*.tar.gz | ||||||
|  |       env: | ||||||
|  |         GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
							
								
								
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -5,3 +5,16 @@ | |||||||
| *.swo | *.swo | ||||||
| *~ | *~ | ||||||
| .DS_Store | .DS_Store | ||||||
|  | my-blog/public/ | ||||||
|  | dist | ||||||
|  | node_modules | ||||||
|  | package-lock.json  | ||||||
|  | my-blog/static/assets/comment-atproto-* | ||||||
|  | bin/ailog | ||||||
|  | docs | ||||||
|  | my-blog/static/index.html | ||||||
|  | my-blog/templates/oauth-assets.html | ||||||
|  | cloudflared-config.yml | ||||||
|  | .config | ||||||
|  | oauth-server-example | ||||||
|  | atproto | ||||||
|   | |||||||
							
								
								
									
										50
									
								
								Cargo.toml
									
									
									
									
									
								
							
							
						
						
									
										50
									
								
								Cargo.toml
									
									
									
									
									
								
							| @@ -1,17 +1,25 @@ | |||||||
| [package] | [package] | ||||||
| name = "ailog" | name = "ailog" | ||||||
| version = "0.1.0" | version = "0.2.1" | ||||||
| edition = "2021" | edition = "2021" | ||||||
| authors = ["syui"] | authors = ["syui"] | ||||||
| description = "A static blog generator with AI features" | description = "A static blog generator with AI features" | ||||||
| license = "MIT" | license = "MIT" | ||||||
|  |  | ||||||
|  | [[bin]] | ||||||
|  | name = "ailog" | ||||||
|  | path = "src/main.rs" | ||||||
|  |  | ||||||
|  | [lib] | ||||||
|  | name = "ailog" | ||||||
|  | path = "src/lib.rs" | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| clap = { version = "4.5", features = ["derive"] } | clap = { version = "4.5", features = ["derive"] } | ||||||
| pulldown-cmark = "0.11" | pulldown-cmark = "0.11" | ||||||
| serde = { version = "1.0", features = ["derive"] } | serde = { version = "1.0", features = ["derive"] } | ||||||
| serde_json = "1.0" | serde_json = "1.0" | ||||||
| tokio = { version = "1.40", features = ["full"] } | tokio = { version = "1.40", features = ["rt-multi-thread", "macros", "fs", "net", "io-util", "sync", "time", "process", "signal"] } | ||||||
| anyhow = "1.0" | anyhow = "1.0" | ||||||
| toml = "0.8" | toml = "0.8" | ||||||
| chrono = "0.4" | chrono = "0.4" | ||||||
| @@ -22,7 +30,7 @@ fs_extra = "1.3" | |||||||
| colored = "2.1" | colored = "2.1" | ||||||
| serde_yaml = "0.9" | serde_yaml = "0.9" | ||||||
| syntect = "5.2" | syntect = "5.2" | ||||||
| reqwest = { version = "0.12", features = ["json"] } | reqwest = { version = "0.12", features = ["json", "rustls-tls"], default-features = false } | ||||||
| rand = "0.8" | rand = "0.8" | ||||||
| sha2 = "0.10" | sha2 = "0.10" | ||||||
| base64 = "0.22" | base64 = "0.22" | ||||||
| @@ -32,12 +40,46 @@ axum = "0.7" | |||||||
| tower = "0.5" | tower = "0.5" | ||||||
| tower-http = { version = "0.5", features = ["cors", "fs"] } | tower-http = { version = "0.5", features = ["cors", "fs"] } | ||||||
| hyper = { version = "1.0", features = ["full"] } | hyper = { version = "1.0", features = ["full"] } | ||||||
|  | tower-sessions = "0.12" | ||||||
|  | jsonwebtoken = "9.2" | ||||||
|  | cookie = "0.18" | ||||||
| # Documentation generation dependencies | # Documentation generation dependencies | ||||||
| syn = { version = "2.0", features = ["full", "parsing", "visit"] } | syn = { version = "2.0", features = ["full", "parsing", "visit"] } | ||||||
| quote = "1.0" | quote = "1.0" | ||||||
| ignore = "0.4" | ignore = "0.4" | ||||||
| git2 = "0.18" | git2 = { version = "0.18", features = ["vendored-openssl", "vendored-libgit2", "ssh"], default-features = false } | ||||||
| regex = "1.0" | regex = "1.0" | ||||||
|  | # ATProto and stream monitoring dependencies | ||||||
|  | tokio-tungstenite = { version = "0.21", features = ["rustls-tls-webpki-roots", "connect"], default-features = false } | ||||||
|  | futures-util = "0.3" | ||||||
|  | tungstenite = { version = "0.21", features = ["rustls-tls-webpki-roots"], default-features = false } | ||||||
|  | rpassword = "7.3" | ||||||
|  |  | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| tempfile = "3.14" | tempfile = "3.14" | ||||||
|  |  | ||||||
|  | [profile.dev] | ||||||
|  | # Speed up development builds | ||||||
|  | opt-level = 0 | ||||||
|  | debug = true | ||||||
|  | debug-assertions = true | ||||||
|  | overflow-checks = true | ||||||
|  | lto = false | ||||||
|  | panic = 'unwind' | ||||||
|  | incremental = true | ||||||
|  | codegen-units = 256 | ||||||
|  |  | ||||||
|  | [profile.release] | ||||||
|  | # Optimize release builds for speed and size | ||||||
|  | opt-level = 3 | ||||||
|  | debug = false | ||||||
|  | debug-assertions = false | ||||||
|  | overflow-checks = false | ||||||
|  | lto = true | ||||||
|  | panic = 'abort' | ||||||
|  | incremental = false | ||||||
|  | codegen-units = 1 | ||||||
|  |  | ||||||
|  | [profile.dev.package."*"] | ||||||
|  | # Optimize dependencies in dev builds | ||||||
|  | opt-level = 3 | ||||||
							
								
								
									
										820
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										820
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,438 +1,556 @@ | |||||||
| # ai.log | # ai.log | ||||||
|  |  | ||||||
| A Rust-based static blog generator with AI integration capabilities. | AI-powered static blog generator with ATProto integration, part of the ai.ai ecosystem. | ||||||
|  |  | ||||||
| ## Overview | ## 🚀 Quick Start | ||||||
|  |  | ||||||
| ai.log is part of the ai ecosystem - a static site generator that creates blogs with built-in AI features for content enhancement and atproto integration. The system follows the yui system principles with dual-layer MCP architecture. | ### Installation & Setup | ||||||
|  |  | ||||||
| ## Architecture |  | ||||||
|  |  | ||||||
| ### Dual MCP Integration |  | ||||||
|  |  | ||||||
| **ai.log MCP Server (API Layer)** |  | ||||||
| - **Role**: Independent blog API |  | ||||||
| - **Port**: 8002 |  | ||||||
| - **Location**: `./src/mcp/` |  | ||||||
| - **Function**: Core blog generation and management |  | ||||||
|  |  | ||||||
| **ai.gpt Integration (Server Layer)** |  | ||||||
| - **Role**: AI integration gateway |  | ||||||
| - **Port**: 8001 (within ai.gpt) |  | ||||||
| - **Location**: `../src/aigpt/mcp_server.py` |  | ||||||
| - **Function**: AI memory system + HTTP proxy to ai.log |  | ||||||
|  |  | ||||||
| ### Data Flow |  | ||||||
| ``` |  | ||||||
| Claude Code → ai.gpt (Server/AI) → ai.log (API/Blog) → Static Site |  | ||||||
|               ↑                      ↑ |  | ||||||
|               Memory System          File Operations |  | ||||||
|               Relationship AI        Markdown Processing |  | ||||||
|               Context Analysis       Template Rendering |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## Features |  | ||||||
|  |  | ||||||
| - **Static Blog Generation**: Inspired by Zola, built with Rust |  | ||||||
| - **AI-Powered Content**: Memory-driven article generation via ai.gpt |  | ||||||
| - **🌍 Ollama Translation**: Multi-language markdown translation with structure preservation |  | ||||||
| - **atproto Integration**: OAuth authentication and comment system (planned) |  | ||||||
| - **MCP Integration**: Seamless Claude Code workflow |  | ||||||
|  |  | ||||||
| ## Installation |  | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| cargo install ailog | # 1. Clone repository | ||||||
|  | git clone https://git.syui.ai/ai/log | ||||||
|  | cd log | ||||||
|  |  | ||||||
|  | # 2. Build ailog | ||||||
|  | cargo build --release | ||||||
|  |  | ||||||
|  | # 3. Initialize blog | ||||||
|  | ./target/release/ailog init my-blog | ||||||
|  |  | ||||||
|  | # 4. Create your first post | ||||||
|  | ./target/release/ailog new "My First Post" | ||||||
|  |  | ||||||
|  | # 5. Build static site | ||||||
|  | ./target/release/ailog build | ||||||
|  |  | ||||||
|  | # 6. Serve locally | ||||||
|  | ./target/release/ailog serve | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Usage | ### Install via Cargo | ||||||
|  |  | ||||||
| ### Standalone Mode |  | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| # Initialize a new blog | cargo install --path . | ||||||
| ailog init myblog | # Now you can use `ailog` command globally | ||||||
|  |  | ||||||
| # Create a new post |  | ||||||
| ailog new "My First Post" |  | ||||||
|  |  | ||||||
| # Build the blog |  | ||||||
| ailog build |  | ||||||
|  |  | ||||||
| # Serve locally |  | ||||||
| ailog serve |  | ||||||
|  |  | ||||||
| # Start MCP server |  | ||||||
| ailog mcp --port 8002 |  | ||||||
|  |  | ||||||
| # Generate documentation |  | ||||||
| ailog doc readme --with-ai |  | ||||||
| ailog doc api --output ./docs |  | ||||||
| ailog doc structure --include-deps |  | ||||||
|  |  | ||||||
| # Translate documents (requires Ollama) |  | ||||||
| ailog doc translate --input README.md --target-lang en |  | ||||||
| ailog doc translate --input docs/api.md --target-lang ja --model qwen2.5:latest |  | ||||||
|  |  | ||||||
| # Clean build files |  | ||||||
| ailog clean |  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### AI Ecosystem Integration | ## 📖 Core Commands | ||||||
|  |  | ||||||
| When integrated with ai.gpt, use natural language: | ### Blog Management | ||||||
| - "ブログ記事を書いて" → Triggers `log_ai_content` |  | ||||||
| - "記事一覧を見せて" → Triggers `log_list_posts` | ```bash | ||||||
| - "ブログをビルドして" → Triggers `log_build_blog` | # Project setup | ||||||
|  | ailog init <project-name>           # Initialize new blog project | ||||||
|  | ailog new <title>                   # Create new blog post | ||||||
|  | ailog build                         # Generate static site with JSON index | ||||||
|  | ailog serve                         # Start development server | ||||||
|  | ailog clean                         # Clean build artifacts | ||||||
|  |  | ||||||
|  | # ATProto authentication   | ||||||
|  | ailog auth init                     # Setup ATProto credentials | ||||||
|  | ailog auth status                   # Check authentication status | ||||||
|  | ailog auth logout                   # Clear credentials | ||||||
|  |  | ||||||
|  | # OAuth app build | ||||||
|  | ailog oauth build <project-dir>     # Build OAuth comment system | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Stream & AI Features | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # Start monitoring & AI generation | ||||||
|  | ailog stream start --ai-generate    # Monitor blog + auto-generate AI content | ||||||
|  | ailog stream start --daemon         # Run as background daemon | ||||||
|  | ailog stream status                 # Check stream status | ||||||
|  | ailog stream stop                   # Stop monitoring | ||||||
|  | ailog stream test                   # Test ATProto API access | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ### Documentation & Translation | ### Documentation & Translation | ||||||
|  |  | ||||||
| Generate comprehensive documentation and translate content: | ```bash | ||||||
| - "READMEを生成して" → Triggers `log_generate_docs` | # Generate documentation | ||||||
| - "APIドキュメントを作成して" → Generates API documentation | ailog doc readme --with-ai          # Generate enhanced README | ||||||
| - "プロジェクト構造を解析して" → Creates structure documentation | ailog doc api --output ./docs       # Generate API documentation | ||||||
| - "このファイルを英語に翻訳して" → Triggers `log_translate_document` | ailog doc structure --include-deps  # Analyze project structure | ||||||
| - "マークダウンを日本語に変換して" → Uses Ollama for translation |  | ||||||
|  |  | ||||||
| ## MCP Tools | # AI-powered translation | ||||||
|  | ailog doc translate --input README.md --target-lang en | ||||||
|  | ailog doc translate --input docs/guide.ja.md --target-lang en --model qwen2.5:latest | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ### ai.log Server (Port 8002) | ## 🏗️ Architecture | ||||||
| - `create_blog_post` - Create new blog post |  | ||||||
| - `list_blog_posts` - List existing posts |  | ||||||
| - `build_blog` - Build static site |  | ||||||
| - `get_post_content` - Get post by slug |  | ||||||
| - `translate_document` ⭐ - Ollama-powered markdown translation |  | ||||||
| - `generate_documentation` ⭐ - Code analysis and documentation generation |  | ||||||
|  |  | ||||||
| ### ai.gpt Integration (Port 8001) | ### Project Structure | ||||||
| - `log_create_post` - Proxy to ai.log + error handling |  | ||||||
| - `log_list_posts` - Proxy to ai.log + formatting |  | ||||||
| - `log_build_blog` - Proxy to ai.log + AI features |  | ||||||
| - `log_get_post` - Proxy to ai.log + context |  | ||||||
| - `log_system_status` - Health check for ai.log |  | ||||||
| - `log_ai_content` ⭐ - AI memory → blog content generation |  | ||||||
| - `log_translate_document` 🌍 - Document translation via Ollama |  | ||||||
| - `log_generate_docs` 📚 - Documentation generation |  | ||||||
|  |  | ||||||
| ### Documentation Generation Tools | ``` | ||||||
| - `doc readme` - Generate README.md from project analysis | ai.log/ | ||||||
| - `doc api` - Generate API documentation | ├── src/                    # Rust static blog generator | ||||||
| - `doc structure` - Analyze and document project structure | │   ├── commands/          # CLI command implementations | ||||||
| - `doc changelog` - Generate changelog from git history | │   ├── generator.rs       # Core blog generation + JSON index | ||||||
| - `doc translate` 🌍 - Multi-language document translation | │   ├── mcp/              # MCP server integration | ||||||
|  | │   └── main.rs           # CLI entry point | ||||||
|  | ├── my-blog/              # Your blog content | ||||||
|  | │   ├── content/posts/    # Markdown blog posts | ||||||
|  | │   ├── templates/        # Tera templates | ||||||
|  | │   ├── static/          # Static assets | ||||||
|  | │   └── public/          # Generated site output | ||||||
|  | ├── oauth/               # ATProto comment system | ||||||
|  | │   ├── src/            # TypeScript OAuth app | ||||||
|  | │   ├── dist/           # Built OAuth assets | ||||||
|  | │   └── package.json    # Node.js dependencies | ||||||
|  | └── target/             # Rust build output | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ### Translation Features | ### Data Flow | ||||||
| - **Language Support**: English, Japanese, Chinese, Korean, Spanish |  | ||||||
| - **Markdown Preservation**: Code blocks, links, images, tables maintained |  | ||||||
| - **Auto-Detection**: Automatically detects Japanese content |  | ||||||
| - **Ollama Integration**: Uses local AI models for privacy and cost-efficiency |  | ||||||
| - **Smart Processing**: Section-by-section translation with structure awareness |  | ||||||
|  |  | ||||||
| ## Configuration | ``` | ||||||
|  | Blog Posts (Markdown) → ailog build → public/ | ||||||
|  |                                    ├── Static HTML pages | ||||||
|  |                                    └── index.json (API) | ||||||
|  |                                          ↓ | ||||||
|  | ailog stream start --ai-generate → Monitor index.json | ||||||
|  |                                          ↓ | ||||||
|  | New posts detected → Ollama AI → ATProto records | ||||||
|  |                                ├── ai.syui.log.chat.lang (translations) | ||||||
|  |                                └── ai.syui.log.chat.comment (AI comments) | ||||||
|  |                                          ↓ | ||||||
|  | OAuth app → Display AI-generated content | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ### ai.log Configuration | ## 🤖 AI Integration | ||||||
| - Location: `~/.config/syui/ai/log/` |  | ||||||
| - Format: TOML configuration |  | ||||||
|  |  | ||||||
| ### ai.gpt Integration | ### AI Content Generation | ||||||
| - Configuration: `../config.json` |  | ||||||
| - Auto-detection: ai.log tools enabled when `./log/` directory exists |  | ||||||
| - System prompt: Automatically triggers blog tools for related queries |  | ||||||
|  |  | ||||||
| ## AI Integration Features | The `--ai-generate` flag enables automatic AI content generation: | ||||||
|  |  | ||||||
| ### Memory-Driven Content Generation | 1. **Blog Monitoring**: Monitors `index.json` every 5 minutes | ||||||
| - **Source**: ai.gpt memory system | 2. **Duplicate Prevention**: Checks existing ATProto collections | ||||||
| - **Process**: Contextual memories → AI analysis → Blog content | 3. **AI Generation**: Uses Ollama (gemma3:4b) for translations & comments | ||||||
| - **Output**: Structured markdown with personal insights | 4. **ATProto Storage**: Saves to derived collections (`base.chat.lang`, `base.chat.comment`) | ||||||
|  |  | ||||||
| ### Automatic Workflows | ```bash | ||||||
| - Daily blog posts from accumulated memories | # Start AI generation monitor | ||||||
| - Content enhancement and suggestions | ailog stream start --ai-generate | ||||||
| - Related article recommendations |  | ||||||
| - Multi-language content generation |  | ||||||
|  |  | ||||||
| ## atproto Integration (Planned) | # Output: | ||||||
|  | # 🤖 Starting AI content generation monitor... | ||||||
|  | # 📡 Blog host: https://syui.ai | ||||||
|  | # 🧠 Ollama host: https://ollama.syui.ai | ||||||
|  | # 🔍 Checking for new blog posts... | ||||||
|  | # ✅ Generated translation for: 静的サイトジェネレータを作った | ||||||
|  | # ✅ Generated comment for: 静的サイトジェネレータを作った | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ### OAuth 2.0 Authentication | ### Collection Management | ||||||
| - Client metadata: `public/client-metadata.json` |  | ||||||
| - Comment system integration | ailog uses a **simplified collection structure** based on a single base collection name: | ||||||
| - Data sovereignty: Users own their comments |  | ||||||
| - Collection storage in atproto | ```bash | ||||||
|  | # Single environment variable controls all collections (unified naming) | ||||||
|  | export VITE_OAUTH_COLLECTION="ai.syui.log" | ||||||
|  |  | ||||||
|  | # Automatically derives: | ||||||
|  | # - ai.syui.log (comments) | ||||||
|  | # - ai.syui.log.user (user management)   | ||||||
|  | # - ai.syui.log.chat.lang (AI translations) | ||||||
|  | # - ai.syui.log.chat.comment (AI comments) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | **Benefits:** | ||||||
|  | - ✅ **Simple**: One variable instead of 5+ | ||||||
|  | - ✅ **Consistent**: All collections follow the same pattern | ||||||
|  | - ✅ **Manageable**: Easy systemd/production configuration | ||||||
|  |  | ||||||
|  | ### Ask AI Feature | ||||||
|  |  | ||||||
|  | Interactive AI chat integrated into blog pages: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # 1. Setup Ollama | ||||||
|  | brew install ollama | ||||||
|  | ollama pull gemma2:2b | ||||||
|  |  | ||||||
|  | # 2. Start with CORS support | ||||||
|  | OLLAMA_ORIGINS="https://example.com" ollama serve | ||||||
|  |  | ||||||
|  | # 3. Configure AI DID in templates/base.html | ||||||
|  | const aiConfig = { | ||||||
|  |     systemPrompt: 'You are a helpful AI assistant.', | ||||||
|  |     aiDid: 'did:plc:your-ai-bot-did' | ||||||
|  | }; | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 🌐 ATProto Integration | ||||||
|  |  | ||||||
|  | ### OAuth Comment System | ||||||
|  |  | ||||||
|  | The OAuth app provides ATProto-authenticated commenting: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # 1. Build OAuth app | ||||||
|  | cd oauth | ||||||
|  | npm install | ||||||
|  | npm run build | ||||||
|  |  | ||||||
|  | # 2. Configure for production | ||||||
|  | ailog oauth build my-blog  # Auto-generates .env.production | ||||||
|  |  | ||||||
|  | # 3. Deploy OAuth assets | ||||||
|  | # Assets are automatically copied to public/ during ailog build | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Authentication Setup | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # Initialize ATProto authentication | ||||||
|  | ailog auth init | ||||||
|  |  | ||||||
|  | # Input required: | ||||||
|  | # - Handle (e.g., your.handle.bsky.social) | ||||||
|  | # - Access JWT | ||||||
|  | # - Refresh JWT | ||||||
|  |  | ||||||
|  | # Check status | ||||||
|  | ailog auth status | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Collection Structure | ||||||
|  |  | ||||||
|  | All ATProto collections are **automatically derived** from a single base name: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | Base Collection: "ai.syui.log" | ||||||
|  | ├── ai.syui.log (user comments) | ||||||
|  | ├── ai.syui.log.user (registered commenters) | ||||||
|  | └── ai.syui.log.chat/ | ||||||
|  |     ├── ai.syui.log.chat.lang (AI translations) | ||||||
|  |     └── ai.syui.log.chat.comment (AI comments) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | **Configuration Priority:** | ||||||
|  | 1. Environment variable: `VITE_OAUTH_COLLECTION` (unified) | ||||||
|  | 2. config.toml: `[oauth] collection = "..."` | ||||||
|  | 3. Auto-generated from domain (e.g., `log.syui.ai` → `ai.syui.log`) | ||||||
|  | 4. Default: `ai.syui.log` | ||||||
|  |  | ||||||
|  | ### Stream Monitoring | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # Monitor ATProto streams for comments | ||||||
|  | ailog stream start | ||||||
|  |  | ||||||
|  | # Enable AI generation alongside monitoring | ||||||
|  | ailog stream start --ai-generate --daemon | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 📱 OAuth App Features | ||||||
|  |  | ||||||
|  | The OAuth TypeScript app provides: | ||||||
|  |  | ||||||
| ### Comment System | ### Comment System | ||||||
| - atproto account login | - **Real-time Comments**: ATProto-authenticated commenting | ||||||
| - Distributed comment storage | - **User Management**: Automatic user registration | ||||||
| - Real-time comment synchronization | - **Mobile Responsive**: Optimized for all devices | ||||||
|  | - **JSON View**: Technical record inspection | ||||||
|  |  | ||||||
| ## Build & Deploy | ### AI Content Display | ||||||
|  | - **Lang: EN Tab**: AI-generated English translations | ||||||
|  | - **AI Comment Tab**: AI-generated blog insights | ||||||
|  | - **Admin Records**: Fetches from admin DID collections | ||||||
|  | - **Real-time Updates**: Live content refresh | ||||||
|  |  | ||||||
| ### GitHub Actions | ### Setup & Configuration | ||||||
| ```yaml |  | ||||||
| # .github/workflows/gh-pages.yml |  | ||||||
| - name: Build ai.log |  | ||||||
|   run: | |  | ||||||
|     cd log |  | ||||||
|     cargo build --release |  | ||||||
|     ./target/release/ailog build |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Cloudflare Pages |  | ||||||
| - Static output: `./public/` |  | ||||||
| - Automatic deployment on main branch push |  | ||||||
| - AI content generation during build process |  | ||||||
|  |  | ||||||
| ## Development Status |  | ||||||
|  |  | ||||||
| ### ✅ Completed Features |  | ||||||
| - Project structure and Cargo.toml setup |  | ||||||
| - CLI interface (init, new, build, serve, clean, mcp, doc) |  | ||||||
| - Configuration system with TOML support |  | ||||||
| - Markdown parsing with frontmatter support |  | ||||||
| - Template system with Handlebars |  | ||||||
| - Static site generation with posts and pages |  | ||||||
| - Development server with hot reload |  | ||||||
| - **MCP server integration (both layers)** |  | ||||||
| - **ai.gpt integration with 6 tools** |  | ||||||
| - **AI memory system connection** |  | ||||||
| - **📚 Documentation generation from code** |  | ||||||
| - **🔍 Rust project analysis and API extraction** |  | ||||||
| - **📝 README, API docs, and structure analysis** |  | ||||||
| - **🌍 Ollama-powered translation system** |  | ||||||
| - **🚀 Complete MCP integration with ai.gpt** |  | ||||||
| - **📄 Markdown-aware translation preserving structure** |  | ||||||
| - Test blog with sample content and styling |  | ||||||
|  |  | ||||||
| ### 🚧 In Progress |  | ||||||
| - AI-powered content enhancement pipeline |  | ||||||
| - atproto OAuth implementation |  | ||||||
|  |  | ||||||
| ### 📋 Planned Features |  | ||||||
| - Advanced template customization |  | ||||||
| - Plugin system for extensibility |  | ||||||
| - Real-time comment system |  | ||||||
| - Multi-blog management |  | ||||||
| - VTuber integration (ai.verse connection) |  | ||||||
|  |  | ||||||
| ## Integration with ai Ecosystem |  | ||||||
|  |  | ||||||
| ### System Dependencies |  | ||||||
| - **ai.gpt**: Memory system, relationship tracking, AI provider |  | ||||||
| - **ai.card**: Future cross-system content sharing |  | ||||||
| - **ai.bot**: atproto posting and mention handling |  | ||||||
| - **ai.verse**: 3D world blog representation (future) |  | ||||||
|  |  | ||||||
| ### yui System Compliance |  | ||||||
| - **Uniqueness**: Each blog post tied to individual identity |  | ||||||
| - **Reality Reflection**: Personal memories → digital content |  | ||||||
| - **Irreversibility**: Published content maintains historical integrity |  | ||||||
|  |  | ||||||
| ## Getting Started |  | ||||||
|  |  | ||||||
| ### 1. Standalone Usage |  | ||||||
| ```bash | ```bash | ||||||
| git clone [repository] | cd oauth | ||||||
| cd log |  | ||||||
| cargo run -- init my-blog | # Development | ||||||
| cargo run -- new "First Post" | npm run dev | ||||||
| cargo run -- build |  | ||||||
| cargo run -- serve | # Production build | ||||||
|  | npm run build | ||||||
|  |  | ||||||
|  | # Preview production | ||||||
|  | npm run preview | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### 2. AI Ecosystem Integration | **Environment Variables:** | ||||||
| ```bash | ```bash | ||||||
| # Start ai.log MCP server | # Production (.env.production - auto-generated by ailog oauth build) | ||||||
| cargo run -- mcp --port 8002 | VITE_APP_HOST=https://syui.ai | ||||||
|  | VITE_OAUTH_CLIENT_ID=https://syui.ai/client-metadata.json | ||||||
|  | VITE_OAUTH_REDIRECT_URI=https://syui.ai/oauth/callback | ||||||
|  | VITE_ADMIN_DID=did:plc:uqzpqmrjnptsxezjx4xuh2mn | ||||||
|  |  | ||||||
| # In another terminal, start ai.gpt | # Simplified collection configuration (single base collection) | ||||||
| cd ../ | VITE_OAUTH_COLLECTION=ai.syui.log | ||||||
| # ai.gpt startup commands |  | ||||||
|  |  | ||||||
| # Use Claude Code with natural language blog commands | # AI Configuration | ||||||
|  | VITE_AI_ENABLED=true | ||||||
|  | VITE_AI_ASK_AI=true | ||||||
|  | VITE_AI_PROVIDER=ollama | ||||||
|  | # ... (other AI settings) | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Documentation Generation Features | ## 🔧 Advanced Features | ||||||
|  |  | ||||||
| ### 📚 Automatic README Generation | ### JSON Index Generation | ||||||
| ```bash |  | ||||||
| # Generate README from project analysis |  | ||||||
| ailog doc readme --source ./src --with-ai |  | ||||||
|  |  | ||||||
| # Output: Enhanced README.md with: | Every `ailog build` generates `/public/index.json`: | ||||||
| # - Project overview and metrics |  | ||||||
| # - Dependency analysis | ```json | ||||||
| # - Module structure | [ | ||||||
| # - AI-generated insights |   { | ||||||
|  |     "title": "静的サイトジェネレータを作った", | ||||||
|  |     "href": "https://syui.ai/posts/2025-06-06-ailog.html", | ||||||
|  |     "formated_time": "Thu Jun 12, 2025", | ||||||
|  |     "utc_time": "2025-06-12T00:00:00Z", | ||||||
|  |     "tags": ["blog", "rust", "mcp", "atp"], | ||||||
|  |     "contents": "Plain text content...", | ||||||
|  |     "description": "Excerpt...", | ||||||
|  |     "categories": [] | ||||||
|  |   } | ||||||
|  | ] | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### 📖 API Documentation | This enables: | ||||||
| ```bash | - **API Access**: Programmatic blog content access | ||||||
| # Generate comprehensive API docs | - **Stream Monitoring**: AI generation triggers | ||||||
| ailog doc api --source ./src --format markdown --output ./docs | - **Search Integration**: Full-text search capabilities | ||||||
|  |  | ||||||
| # Creates: | ### Translation System | ||||||
| # - docs/api.md (main API overview) |  | ||||||
| # - docs/module_name.md (per-module documentation) |  | ||||||
| # - Function signatures and documentation |  | ||||||
| # - Struct/enum definitions |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### 🏗️ Project Structure Analysis | AI-powered document translation with Ollama: | ||||||
| ```bash |  | ||||||
| # Analyze and document project structure |  | ||||||
| ailog doc structure --source . --include-deps |  | ||||||
|  |  | ||||||
| # Generates: |  | ||||||
| # - Directory tree visualization |  | ||||||
| # - File distribution by language |  | ||||||
| # - Dependency graph analysis |  | ||||||
| # - Code metrics and statistics |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### 📝 Git Changelog Generation |  | ||||||
| ```bash |  | ||||||
| # Generate changelog from git history |  | ||||||
| ailog doc changelog --from v1.0.0 --explain-changes |  | ||||||
|  |  | ||||||
| # Creates: |  | ||||||
| # - Structured changelog |  | ||||||
| # - Commit categorization |  | ||||||
| # - AI-enhanced change explanations |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### 🤖 AI-Enhanced Documentation |  | ||||||
| When `--with-ai` is enabled: |  | ||||||
| - **Content Enhancement**: AI improves readability and adds insights |  | ||||||
| - **Context Awareness**: Leverages ai.gpt memory system |  | ||||||
| - **Smart Categorization**: Automatic organization of content |  | ||||||
| - **Technical Writing**: Professional documentation style |  | ||||||
|  |  | ||||||
| ## 🌍 Translation System |  | ||||||
|  |  | ||||||
| ### Ollama-Powered Translation |  | ||||||
|  |  | ||||||
| ai.log includes a comprehensive translation system powered by Ollama AI models: |  | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| # Basic translation | # Basic translation | ||||||
| ailog doc translate --input README.md --target-lang en | ailog doc translate --input README.md --target-lang en | ||||||
|  |  | ||||||
| # Advanced translation with custom settings | # Advanced options | ||||||
| ailog doc translate \ | ailog doc translate \ | ||||||
|   --input docs/technical-guide.ja.md \ |   --input docs/guide.ja.md \ | ||||||
|   --target-lang en \ |   --target-lang en \ | ||||||
|   --source-lang ja \ |   --source-lang ja \ | ||||||
|   --output docs/technical-guide.en.md \ |  | ||||||
|   --model qwen2.5:latest \ |   --model qwen2.5:latest \ | ||||||
|   --ollama-endpoint http://localhost:11434 |   --output docs/guide.en.md | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Translation Features | **Features:** | ||||||
|  | - **Markdown-aware**: Preserves code blocks, links, tables | ||||||
|  | - **Multiple models**: qwen2.5, gemma3, etc. | ||||||
|  | - **Auto-detection**: Detects Japanese content automatically | ||||||
|  | - **Structure preservation**: Maintains document formatting | ||||||
|  |  | ||||||
| #### 📄 Markdown-Aware Processing | ### MCP Server Integration | ||||||
| - **Code Block Preservation**: All code snippets remain untranslated |  | ||||||
| - **Link Maintenance**: URLs and link structures preserved |  | ||||||
| - **Image Handling**: Alt text can be translated while preserving image paths |  | ||||||
| - **Table Translation**: Table content translated while maintaining structure |  | ||||||
| - **Header Preservation**: Markdown headers translated with level maintenance |  | ||||||
|  |  | ||||||
| #### 🎯 Smart Language Detection | ```bash | ||||||
| - **Auto-Detection**: Automatically detects Japanese content using Unicode ranges | # Start MCP server for ai.gpt integration | ||||||
| - **Manual Override**: Specify source language for precise control | ailog mcp --port 8002 | ||||||
| - **Mixed Content**: Handles documents with multiple languages |  | ||||||
|  |  | ||||||
| #### 🔧 Flexible Configuration | # Available tools: | ||||||
| - **Model Selection**: Choose from available Ollama models | # - create_blog_post | ||||||
| - **Custom Endpoints**: Use different Ollama instances | # - list_blog_posts   | ||||||
| - **Output Control**: Auto-generate or specify output paths | # - build_blog | ||||||
| - **Batch Processing**: Process multiple files efficiently | # - get_post_content | ||||||
|  | # - translate_document | ||||||
|  | # - generate_documentation | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 🚀 Deployment | ||||||
|  |  | ||||||
|  | ### GitHub Actions | ||||||
|  |  | ||||||
|  | ```yaml | ||||||
|  | name: Deploy ai.log Blog | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [main] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   deploy: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |      | ||||||
|  |     - name: Setup Rust | ||||||
|  |       uses: actions-rs/toolchain@v1 | ||||||
|  |       with: | ||||||
|  |         toolchain: stable | ||||||
|  |          | ||||||
|  |     - name: Build ailog | ||||||
|  |       run: cargo build --release | ||||||
|  |        | ||||||
|  |     - name: Build blog | ||||||
|  |       run: | | ||||||
|  |         cd my-blog | ||||||
|  |         ../target/release/ailog build | ||||||
|  |          | ||||||
|  |     - name: Deploy to Cloudflare Pages | ||||||
|  |       uses: cloudflare/pages-action@v1 | ||||||
|  |       with: | ||||||
|  |         apiToken: ${{ secrets.CLOUDFLARE_API_TOKEN }} | ||||||
|  |         accountId: ${{ secrets.CLOUDFLARE_ACCOUNT_ID }} | ||||||
|  |         projectName: my-blog | ||||||
|  |         directory: my-blog/public | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Production Setup | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # 1. Build for production | ||||||
|  | cargo build --release | ||||||
|  |  | ||||||
|  | # 2. Setup systemd services | ||||||
|  | sudo cp systemd/system/ailog-stream.service /etc/systemd/system/ | ||||||
|  | sudo systemctl enable ailog-stream.service | ||||||
|  | sudo systemctl start ailog-stream.service | ||||||
|  |  | ||||||
|  | # 3. Configure Ollama with CORS | ||||||
|  | sudo vim /usr/lib/systemd/system/ollama.service | ||||||
|  | # Add: Environment="OLLAMA_ORIGINS=https://yourdomain.com" | ||||||
|  |  | ||||||
|  | # 4. Monitor services | ||||||
|  | journalctl -u ailog-stream.service -f | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 🌍 Translation Support | ||||||
|  |  | ||||||
| ### Supported Languages | ### Supported Languages | ||||||
|  |  | ||||||
| | Language | Code | Direction | Model Optimized | | | Language | Code | Status | Model | | ||||||
| |----------|------|-----------|-----------------| | |----------|------|--------|-------| | ||||||
| | English  | `en` | ↔️        | ✅ qwen2.5      | | | English  | `en` | ✅ Full | qwen2.5 | | ||||||
| | Japanese | `ja` | ↔️        | ✅ qwen2.5      | | | Japanese | `ja` | ✅ Full | qwen2.5 | | ||||||
| | Chinese  | `zh` | ↔️        | ✅ qwen2.5      | | | Chinese  | `zh` | ✅ Full | qwen2.5 | | ||||||
| | Korean   | `ko` | ↔️        | ⚠️ Basic       | | | Korean   | `ko` | ⚠️ Basic | qwen2.5 | | ||||||
| | Spanish  | `es` | ↔️        | ⚠️ Basic       | | | Spanish  | `es` | ⚠️ Basic | qwen2.5 | | ||||||
|  |  | ||||||
| ### Translation Workflow | ### Translation Workflow | ||||||
|  |  | ||||||
| 1. **Parse Document**: Analyze markdown structure and identify sections | 1. **Parse**: Analyze markdown structure | ||||||
| 2. **Preserve Code**: Isolate code blocks and technical content | 2. **Preserve**: Isolate code blocks and technical content   | ||||||
| 3. **Translate Content**: Process text sections with Ollama AI | 3. **Translate**: Process with Ollama AI | ||||||
| 4. **Reconstruct**: Rebuild document maintaining original formatting | 4. **Reconstruct**: Rebuild with original formatting | ||||||
| 5. **Validate**: Ensure structural integrity and completeness | 5. **Validate**: Ensure structural integrity | ||||||
|  |  | ||||||
| ### Integration with ai.gpt | ## 🎯 Use Cases | ||||||
|  |  | ||||||
| ```python | ### Personal Blog | ||||||
| # Via ai.gpt MCP tools | - **AI-Enhanced**: Automatic translations and AI insights | ||||||
| await log_translate_document( | - **Distributed Comments**: ATProto-based social interaction | ||||||
|     input_file="README.ja.md", | - **Mobile-First**: Responsive OAuth comment system | ||||||
|     target_lang="en", |  | ||||||
|     model="qwen2.5:latest" | ### Technical Documentation | ||||||
| ) | - **Code Analysis**: Automatic API documentation | ||||||
|  | - **Multi-language**: AI-powered translation | ||||||
|  | - **Structure Analysis**: Project overview generation | ||||||
|  |  | ||||||
|  | ### AI Ecosystem Integration | ||||||
|  | - **ai.gpt Connection**: Memory-driven content generation | ||||||
|  | - **MCP Integration**: Claude Code workflow support | ||||||
|  | - **Distributed Identity**: ATProto authentication | ||||||
|  |  | ||||||
|  | ## 🔍 Troubleshooting | ||||||
|  |  | ||||||
|  | ### Build Issues | ||||||
|  | ```bash | ||||||
|  | # Check Rust version | ||||||
|  | rustc --version | ||||||
|  |  | ||||||
|  | # Update dependencies | ||||||
|  | cargo update | ||||||
|  |  | ||||||
|  | # Clean build | ||||||
|  | cargo clean && cargo build --release | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Requirements | ### Authentication Problems | ||||||
|  | ```bash | ||||||
|  | # Reset authentication | ||||||
|  | ailog auth logout | ||||||
|  | ailog auth init | ||||||
|  |  | ||||||
| - **Ollama**: Install and run Ollama locally | # Test API access | ||||||
| - **Models**: Download supported models (qwen2.5:latest recommended) | ailog stream test | ||||||
| - **Memory**: Sufficient RAM for model inference | ``` | ||||||
| - **Network**: For initial model download only |  | ||||||
|  |  | ||||||
| ## Configuration Examples | ### AI Generation Issues | ||||||
|  | ```bash | ||||||
|  | # Check Ollama status | ||||||
|  | curl http://localhost:11434/api/tags | ||||||
|  |  | ||||||
| ### Basic Blog Config | # Test with manual request | ||||||
|  | curl -X POST http://localhost:11434/api/generate \ | ||||||
|  |   -d '{"model":"gemma3:4b","prompt":"Test","stream":false}' | ||||||
|  |  | ||||||
|  | # Check CORS settings | ||||||
|  | # Ensure OLLAMA_ORIGINS includes your domain | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### OAuth App Issues | ||||||
|  | ```bash | ||||||
|  | # Rebuild OAuth assets | ||||||
|  | cd oauth | ||||||
|  | rm -rf dist/ | ||||||
|  | npm run build | ||||||
|  |  | ||||||
|  | # Check environment variables | ||||||
|  | cat .env.production | ||||||
|  |  | ||||||
|  | # Verify client-metadata.json | ||||||
|  | curl https://yourdomain.com/client-metadata.json | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 📚 Documentation | ||||||
|  |  | ||||||
|  | ### Core Concepts | ||||||
|  | - **Static Generation**: Rust-powered site building | ||||||
|  | - **JSON Index**: API-compatible blog data | ||||||
|  | - **ATProto Integration**: Distributed social features | ||||||
|  | - **AI Enhancement**: Automatic content generation | ||||||
|  |  | ||||||
|  | ### File Structure | ||||||
|  | - `config.toml`: Blog configuration (simplified collection setup) | ||||||
|  | - `content/posts/*.md`: Blog post sources | ||||||
|  | - `templates/*.html`: Tera template files | ||||||
|  | - `public/`: Generated static site + API (index.json) | ||||||
|  | - `oauth/dist/`: Built OAuth assets | ||||||
|  |  | ||||||
|  | ### Example config.toml | ||||||
| ```toml | ```toml | ||||||
| [blog] | [site] | ||||||
| title = "My AI Blog" | title = "My Blog" | ||||||
| description = "Personal thoughts and AI insights" | base_url = "https://myblog.com" | ||||||
| base_url = "https://myblog.example.com" |  | ||||||
|  | [oauth] | ||||||
|  | admin = "did:plc:your-admin-did" | ||||||
|  | collection = "ai.myblog.log"  # Single base collection | ||||||
|  |  | ||||||
| [ai] | [ai] | ||||||
| provider = "openai" | enabled = true | ||||||
| model = "gpt-4" | auto_translate = true | ||||||
| translation = true | comment_moderation = true | ||||||
|  | model = "gemma3:4b" | ||||||
|  | host = "https://ollama.syui.ai" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Advanced Integration | ## 🔗 ai.ai Ecosystem | ||||||
| ```json |  | ||||||
| // ../config.json (ai.gpt) |  | ||||||
| { |  | ||||||
|   "mcp": { |  | ||||||
|     "servers": { |  | ||||||
|       "ai_gpt": { |  | ||||||
|         "endpoints": { |  | ||||||
|           "log_ai_content": "/log_ai_content", |  | ||||||
|           "log_create_post": "/log_create_post" |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## Troubleshooting | ai.log is part of the broader ai.ai ecosystem: | ||||||
|  |  | ||||||
| ### MCP Connection Issues | - **ai.gpt**: Memory system and AI integration | ||||||
| - Ensure ai.log server is running: `cargo run -- mcp --port 8002` | - **ai.card**: ATProto-based card game system   | ||||||
| - Check ai.gpt config includes log endpoints | - **ai.bot**: Social media automation | ||||||
| - Verify `./log/` directory exists relative to ai.gpt | - **ai.verse**: 3D virtual world integration | ||||||
|  | - **ai.shell**: AI-powered shell interface | ||||||
|  |  | ||||||
| ### Build Failures | ### yui System Compliance | ||||||
| - Check Rust version: `rustc --version` | - **Uniqueness**: Each blog tied to individual identity | ||||||
| - Update dependencies: `cargo update` | - **Reality Reflection**: Personal memories → digital content | ||||||
| - Clear cache: `cargo clean` | - **Irreversibility**: Published content maintains integrity | ||||||
|  |  | ||||||
| ### AI Integration Problems | ## 📝 License | ||||||
| - Verify ai.gpt memory system is initialized |  | ||||||
| - Check AI provider configuration |  | ||||||
| - Ensure sufficient context in memory system |  | ||||||
|  |  | ||||||
| ## License |  | ||||||
|  |  | ||||||
| © syui | © syui | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								bin/ailog-linux-x86_64.tar.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								bin/ailog-linux-x86_64.tar.gz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										222
									
								
								claude.md
									
									
									
									
									
								
							
							
						
						
									
										222
									
								
								claude.md
									
									
									
									
									
								
							| @@ -1,5 +1,227 @@ | |||||||
| # エコシステム統合設計書 | # エコシステム統合設計書 | ||||||
|  |  | ||||||
|  | ## 注意事項 | ||||||
|  |  | ||||||
|  | `console.log`は絶対に書かないようにしてください。 | ||||||
|  |  | ||||||
|  | ハードコードしないようにしてください。必ず、`./my-blog/config.toml`や`./oauth/.env.production`を使用するように。または`~/.config/syui/ai/log/config.json`を使用するように。 | ||||||
|  |  | ||||||
|  | 重複する名前のenvを作らないようにしてください。新しい環境変数を作る際は必ず検討してください。 | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | # ダメな例 | ||||||
|  | VITE_OAUTH_COLLECTION_USER=ai.syui.log.user | ||||||
|  | VITE_OAUTH_COLLECTION_CHAT=ai.syui.log.chat | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## oauth appの設計 | ||||||
|  |  | ||||||
|  | > ./oauth/.env.production | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | VITE_ATPROTO_PDS=syu.is | ||||||
|  | VITE_ADMIN_HANDLE=ai.syui.ai | ||||||
|  | VITE_AI_HANDLE=ai.syui.ai | ||||||
|  | VITE_OAUTH_COLLECTION=ai.syui.log | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | これらは非常にシンプルな流れになっており、すべての項目は、共通します。短縮できる場合があります。handleは変わる可能性があるので、できる限りdidを使いましょう。 | ||||||
|  |  | ||||||
|  | 1. handleからpds, didを取得できる ... com.atproto.repo.describeRepo | ||||||
|  | 2. pdsが分かれば、pdsApi, bskyApi, plcApiを割り当てられる | ||||||
|  | 3. bskyApiが分かれば、getProfileでavatar-uriを取得できる ... app.bsky.actor.getProfile | ||||||
|  | 4. pdsAPiからアカウントにあるcollectionのrecordの情報を取得できる ... com.atproto.repo.listRecords | ||||||
|  |  | ||||||
|  | ### コメントを表示する | ||||||
|  |  | ||||||
|  | 1. VITE_ADMIN_HANDLEから管理者のhandleを取得する。 | ||||||
|  | 2. VITE_ATPROTO_PDSから管理者のアカウントのpdsを取得する。 | ||||||
|  | 3. pdsからpdsApi, bskApi, plcApiを割り当てる。 | ||||||
|  |  | ||||||
|  | ```rust | ||||||
|  |     match pds { | ||||||
|  |         "bsky.social" | "bsky.app" => NetworkConfig { | ||||||
|  |             pds_api: format!("https://{}", pds), | ||||||
|  |             plc_api: "https://plc.directory".to_string(), | ||||||
|  |             bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |             web_url: "https://bsky.app".to_string(), | ||||||
|  |         }, | ||||||
|  |         "syu.is" => NetworkConfig { | ||||||
|  |             pds_api: "https://syu.is".to_string(), | ||||||
|  |             plc_api: "https://plc.syu.is".to_string(), | ||||||
|  |             bsky_api: "https://bsky.syu.is".to_string(), | ||||||
|  |             web_url: "https://web.syu.is".to_string(), | ||||||
|  |         }, | ||||||
|  |         _ => { | ||||||
|  |             // Default to Bluesky network for unknown PDS | ||||||
|  |             NetworkConfig { | ||||||
|  |                 pds_api: format!("https://{}", pds), | ||||||
|  |                 plc_api: "https://plc.directory".to_string(), | ||||||
|  |                 bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |                 web_url: "https://bsky.app".to_string(), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 4. 管理者アカウントであるVITE_ADMIN_HANDLEとVITE_ATPROTO_PDSから`ai.syui.log.user`というuserlistを取得する。 | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | curl -sL "https://${VITE_ATPROTO_PDS}/xrpc/com.atproto.repo.listRecords?repo=${VITE_ADMIN_HANDLE}&collection=ai.syui.log.user"  | ||||||
|  | --- | ||||||
|  | syui.ai | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 5. ユーザーがわかったら、そのユーザーのpdsを判定する。 | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | curl -sL "https://bsky.social/xrpc/com.atproto.repo.describeRepo?repo=syui.ai" |jq -r ".didDoc.service.[].serviceEndpoint" | ||||||
|  | --- | ||||||
|  | https://shiitake.us-east.host.bsky.network | ||||||
|  |  | ||||||
|  | curl -sL "https://bsky.social/xrpc/com.atproto.repo.describeRepo?repo=syui.ai" |jq -r ".did" | ||||||
|  | --- | ||||||
|  | did:plc:uqzpqmrjnptsxezjx4xuh2mn | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 6. pdsからpdsApi, bskApi, plcApiを割り当てる。 | ||||||
|  |  | ||||||
|  | ```rust | ||||||
|  |     match pds { | ||||||
|  |         "bsky.social" | "bsky.app" => NetworkConfig { | ||||||
|  |             pds_api: format!("https://{}", pds), | ||||||
|  |             plc_api: "https://plc.directory".to_string(), | ||||||
|  |             bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |             web_url: "https://bsky.app".to_string(), | ||||||
|  |         }, | ||||||
|  |         "syu.is" => NetworkConfig { | ||||||
|  |             pds_api: "https://syu.is".to_string(), | ||||||
|  |             plc_api: "https://plc.syu.is".to_string(), | ||||||
|  |             bsky_api: "https://bsky.syu.is".to_string(), | ||||||
|  |             web_url: "https://web.syu.is".to_string(), | ||||||
|  |         }, | ||||||
|  |         _ => { | ||||||
|  |             // Default to Bluesky network for unknown PDS | ||||||
|  |             NetworkConfig { | ||||||
|  |                 pds_api: format!("https://{}", pds), | ||||||
|  |                 plc_api: "https://plc.directory".to_string(), | ||||||
|  |                 bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |                 web_url: "https://bsky.app".to_string(), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 7. ユーザーの情報を取得、表示する | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | bsky_api=https://public.api.bsky.app | ||||||
|  | user_did=did:plc:uqzpqmrjnptsxezjx4xuh2mn | ||||||
|  | curl -sL "$bsky_api/xrpc/app.bsky.actor.getProfile?actor=$user_did"|jq -r .avatar | ||||||
|  | --- | ||||||
|  | https://cdn.bsky.app/img/avatar/plain/did:plc:uqzpqmrjnptsxezjx4xuh2mn/bafkreid6kcc5pnn4b3ar7mj6vi3eiawhxgkcrw3edgbqeacyrlnlcoetea@jpeg | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### AIの情報を表示する | ||||||
|  |  | ||||||
|  | AIが持つ`ai.syui.log.chat.lang`, `ai.syui.log.chat.comment`を表示します。 | ||||||
|  |  | ||||||
|  | なお、これは通常、`VITE_ADMIN_HANDLE`にputRecordされます。そこから情報を読み込みます。`VITE_AI_HANDLE`はそのrecordの`author`のところに入ります。 | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | "author": { | ||||||
|  |   "did": "did:plc:4hqjfn7m6n5hno3doamuhgef", | ||||||
|  |   "avatar": "https://cdn.bsky.app/img/avatar/plain/did:plc:4hqjfn7m6n5hno3doamuhgef/bafkreiaxkv624mffw3cfyi67ufxtwuwsy2mjw2ygezsvtd44ycbgkfdo2a@jpeg", | ||||||
|  |   "handle": "yui.syui.ai", | ||||||
|  |   "displayName": "ai" | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 1. VITE_ADMIN_HANDLEから管理者のhandleを取得する。 | ||||||
|  | 2. VITE_ATPROTO_PDSから管理者のアカウントのpdsを取得する。 | ||||||
|  | 3. pdsからpdsApi, bskApi, plcApiを割り当てる。 | ||||||
|  |  | ||||||
|  | ```rust | ||||||
|  |     match pds { | ||||||
|  |         "bsky.social" | "bsky.app" => NetworkConfig { | ||||||
|  |             pds_api: format!("https://{}", pds), | ||||||
|  |             plc_api: "https://plc.directory".to_string(), | ||||||
|  |             bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |             web_url: "https://bsky.app".to_string(), | ||||||
|  |         }, | ||||||
|  |         "syu.is" => NetworkConfig { | ||||||
|  |             pds_api: "https://syu.is".to_string(), | ||||||
|  |             plc_api: "https://plc.syu.is".to_string(), | ||||||
|  |             bsky_api: "https://bsky.syu.is".to_string(), | ||||||
|  |             web_url: "https://web.syu.is".to_string(), | ||||||
|  |         }, | ||||||
|  |         _ => { | ||||||
|  |             // Default to Bluesky network for unknown PDS | ||||||
|  |             NetworkConfig { | ||||||
|  |                 pds_api: format!("https://{}", pds), | ||||||
|  |                 plc_api: "https://plc.directory".to_string(), | ||||||
|  |                 bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |                 web_url: "https://bsky.app".to_string(), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 4. 管理者アカウントであるVITE_ADMIN_HANDLEとVITE_ATPROTO_PDSから`ai.syui.log.chat.lang`, `ai.syui.log.chat.comment`を取得する。 | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | curl -sL "https://${VITE_ATPROTO_PDS}/xrpc/com.atproto.repo.listRecords?repo=${VITE_ADMIN_HANDLE}&collection=ai.syui.log.chat.comment"  | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 5. AIのprofileを取得する。 | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | curl -sL "https://${VITE_ATPROTO_PDS}/xrpc/com.atproto.repo.describeRepo?repo=$VITE_AI_HANDLE" |jq -r ".didDoc.service.[].serviceEndpoint" | ||||||
|  | --- | ||||||
|  | https://syu.is | ||||||
|  |  | ||||||
|  | curl -sL "https://${VITE_ATPROTO_PDS}/xrpc/com.atproto.repo.describeRepo?repo=$VITE_AI_HANDLE" |jq -r ".did" | ||||||
|  | did:plc:6qyecktefllvenje24fcxnie | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 6. pdsからpdsApi, bskApi, plcApiを割り当てる。 | ||||||
|  |  | ||||||
|  | ```rust | ||||||
|  |     match pds { | ||||||
|  |         "bsky.social" | "bsky.app" => NetworkConfig { | ||||||
|  |             pds_api: format!("https://{}", pds), | ||||||
|  |             plc_api: "https://plc.directory".to_string(), | ||||||
|  |             bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |             web_url: "https://bsky.app".to_string(), | ||||||
|  |         }, | ||||||
|  |         "syu.is" => NetworkConfig { | ||||||
|  |             pds_api: "https://syu.is".to_string(), | ||||||
|  |             plc_api: "https://plc.syu.is".to_string(), | ||||||
|  |             bsky_api: "https://bsky.syu.is".to_string(), | ||||||
|  |             web_url: "https://web.syu.is".to_string(), | ||||||
|  |         }, | ||||||
|  |         _ => { | ||||||
|  |             // Default to Bluesky network for unknown PDS | ||||||
|  |             NetworkConfig { | ||||||
|  |                 pds_api: format!("https://{}", pds), | ||||||
|  |                 plc_api: "https://plc.directory".to_string(), | ||||||
|  |                 bsky_api: "https://public.api.bsky.app".to_string(), | ||||||
|  |                 web_url: "https://bsky.app".to_string(), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 7. AIの情報を取得、表示する | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | bsky_api=https://bsky.syu.is | ||||||
|  | user_did=did:plc:6qyecktefllvenje24fcxnie | ||||||
|  | curl -sL "$bsky_api/xrpc/app.bsky.actor.getProfile?actor=$user_did"|jq -r .avatar | ||||||
|  | --- | ||||||
|  | https://bsky.syu.is/img/avatar/plain/did:plc:6qyecktefllvenje24fcxnie/bafkreiet4pwlnshk7igra5flf2fuxpg2bhvf2apts4rqwcr56hzhgycii4@jpeg | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## 中核思想 | ## 中核思想 | ||||||
| - **存在子理論**: この世界で最も小さいもの(存在子/ai)の探求 | - **存在子理論**: この世界で最も小さいもの(存在子/ai)の探求 | ||||||
| - **唯一性原則**: 現実の個人の唯一性をすべてのシステムで担保 | - **唯一性原則**: 現実の個人の唯一性をすべてのシステムで担保 | ||||||
|   | |||||||
| @@ -1,164 +0,0 @@ | |||||||
| # ai.log MCP Integration Guide |  | ||||||
|  |  | ||||||
| ai.logをai.gptと連携するためのMCPサーバー設定ガイド |  | ||||||
|  |  | ||||||
| ## MCPサーバー起動 |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| # ai.logプロジェクトディレクトリで |  | ||||||
| ./target/debug/ailog mcp --port 8002 |  | ||||||
|  |  | ||||||
| # またはサブディレクトリから |  | ||||||
| ./target/debug/ailog mcp --port 8002 --path /path/to/blog |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## ai.gptでの設定 |  | ||||||
|  |  | ||||||
| ai.logツールはai.gptのMCPサーバーに統合済みです。`config.json`に以下の設定が含まれています: |  | ||||||
|  |  | ||||||
| ```json |  | ||||||
| { |  | ||||||
|   "mcp": { |  | ||||||
|     "enabled": "true", |  | ||||||
|     "auto_detect": "true", |  | ||||||
|     "servers": { |  | ||||||
|       "ai_gpt": { |  | ||||||
|         "base_url": "http://localhost:8001", |  | ||||||
|         "endpoints": { |  | ||||||
|           "log_create_post": "/log_create_post", |  | ||||||
|           "log_list_posts": "/log_list_posts", |  | ||||||
|           "log_build_blog": "/log_build_blog", |  | ||||||
|           "log_get_post": "/log_get_post", |  | ||||||
|           "log_system_status": "/log_system_status", |  | ||||||
|           "log_ai_content": "/log_ai_content" |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| **重要**: ai.logツールを使用するには、ai.logディレクトリが `./log/` に存在し、ai.logのMCPサーバーがポート8002で稼働している必要があります。 |  | ||||||
|  |  | ||||||
| ## 利用可能なMCPツール(ai.gpt統合版) |  | ||||||
|  |  | ||||||
| ### 1. log_create_post |  | ||||||
| 新しいブログ記事を作成します。 |  | ||||||
|  |  | ||||||
| **パラメータ**: |  | ||||||
| - `title` (必須): 記事のタイトル |  | ||||||
| - `content` (必須): Markdown形式の記事内容 |  | ||||||
| - `tags` (オプション): 記事のタグ配列 |  | ||||||
| - `slug` (オプション): カスタムURL slug |  | ||||||
|  |  | ||||||
| **使用例**: |  | ||||||
| ```python |  | ||||||
| # Claude Code/ai.gptから自動呼び出し |  | ||||||
| # "ブログ記事を書いて"という発言で自動トリガー |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### 2. log_list_posts |  | ||||||
| 既存のブログ記事一覧を取得します。 |  | ||||||
|  |  | ||||||
| **パラメータ**: |  | ||||||
| - `limit` (オプション): 取得件数上限 (デフォルト: 10) |  | ||||||
| - `offset` (オプション): スキップ件数 (デフォルト: 0) |  | ||||||
|  |  | ||||||
| ### 3. log_build_blog |  | ||||||
| ブログをビルドして静的ファイルを生成します。 |  | ||||||
|  |  | ||||||
| **パラメータ**: |  | ||||||
| - `enable_ai` (オプション): AI機能を有効化 (デフォルト: true) |  | ||||||
| - `translate` (オプション): 自動翻訳を有効化 (デフォルト: false) |  | ||||||
|  |  | ||||||
| ### 4. log_get_post |  | ||||||
| 指定したスラッグの記事内容を取得します。 |  | ||||||
|  |  | ||||||
| **パラメータ**: |  | ||||||
| - `slug` (必須): 記事のスラッグ |  | ||||||
|  |  | ||||||
| ### 5. log_system_status |  | ||||||
| ai.logシステムの状態を確認します。 |  | ||||||
|  |  | ||||||
| ### 6. log_ai_content ⭐ NEW |  | ||||||
| AI記憶システムと連携して自動でブログ記事を生成・投稿します。 |  | ||||||
|  |  | ||||||
| **パラメータ**: |  | ||||||
| - `user_id` (必須): ユーザーID |  | ||||||
| - `topic` (オプション): 記事のトピック (デフォルト: "daily thoughts") |  | ||||||
|  |  | ||||||
| **機能**: |  | ||||||
| - ai.gptの記憶システムから関連する思い出を取得 |  | ||||||
| - AI技術で記憶をブログ記事に変換 |  | ||||||
| - 自動でai.logに投稿 |  | ||||||
|  |  | ||||||
| ## ai.gptからの連携パターン |  | ||||||
|  |  | ||||||
| ### 記事の自動投稿 |  | ||||||
| ```python |  | ||||||
| # 記憶システムから関連情報を取得 |  | ||||||
| memories = get_contextual_memories("ブログ") |  | ||||||
|  |  | ||||||
| # AI記事生成 |  | ||||||
| content = generate_blog_content(memories) |  | ||||||
|  |  | ||||||
| # ai.logに投稿 |  | ||||||
| result = await mcp_client.call_tool("create_blog_post", { |  | ||||||
|     "title": "今日の思考メモ", |  | ||||||
|     "content": content, |  | ||||||
|     "tags": ["日記", "AI"] |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| # ビルド実行 |  | ||||||
| await mcp_client.call_tool("build_blog", {"enable_ai": True}) |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### 記事一覧の確認と編集 |  | ||||||
| ```python |  | ||||||
| # 記事一覧取得 |  | ||||||
| posts = await mcp_client.call_tool("list_blog_posts", {"limit": 5}) |  | ||||||
|  |  | ||||||
| # 特定記事の内容取得 |  | ||||||
| content = await mcp_client.call_tool("get_post_content", { |  | ||||||
|     "slug": "ai-integration" |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| # 修正版を投稿(上書き) |  | ||||||
| updated_content = enhance_content(content) |  | ||||||
| await mcp_client.call_tool("create_blog_post", { |  | ||||||
|     "title": "AI統合の新しい可能性(改訂版)", |  | ||||||
|     "content": updated_content, |  | ||||||
|     "slug": "ai-integration-revised" |  | ||||||
| }) |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## 自動化ワークフロー |  | ||||||
|  |  | ||||||
| ai.gptのスケジューラーと組み合わせて: |  | ||||||
|  |  | ||||||
| 1. **日次ブログ投稿**: 蓄積された記憶から記事を自動生成・投稿 |  | ||||||
| 2. **記事修正**: 既存記事の内容を自動改善 |  | ||||||
| 3. **関連記事提案**: 過去記事との関連性に基づく新記事提案 |  | ||||||
| 4. **多言語対応**: 自動翻訳によるグローバル展開 |  | ||||||
|  |  | ||||||
| ## エラーハンドリング |  | ||||||
|  |  | ||||||
| MCPツール呼び出し時のエラーは以下の形式で返されます: |  | ||||||
|  |  | ||||||
| ```json |  | ||||||
| { |  | ||||||
|   "jsonrpc": "2.0", |  | ||||||
|   "id": "request_id", |  | ||||||
|   "error": { |  | ||||||
|     "code": -32000, |  | ||||||
|     "message": "エラーメッセージ", |  | ||||||
|     "data": null |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ## セキュリティ考慮事項 |  | ||||||
|  |  | ||||||
| - MCPサーバーはローカルホストでのみ動作 |  | ||||||
| - ai.gptからの認証済みリクエストのみ処理 |  | ||||||
| - ファイルアクセスは指定されたブログディレクトリ内に制限 |  | ||||||
							
								
								
									
										33
									
								
								my-blog/config.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								my-blog/config.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | [site] | ||||||
|  | title = "syui.ai" | ||||||
|  | description = "a blog powered by ailog" | ||||||
|  | base_url = "https://syui.ai" | ||||||
|  | language = "ja" | ||||||
|  | author = "syui" | ||||||
|  |  | ||||||
|  | [build] | ||||||
|  | highlight_code = true | ||||||
|  | highlight_theme = "Monokai" | ||||||
|  | minify = false | ||||||
|  |  | ||||||
|  | [ai] | ||||||
|  | enabled = true | ||||||
|  | auto_translate = false | ||||||
|  | comment_moderation = false | ||||||
|  | ask_ai = true | ||||||
|  | provider = "ollama" | ||||||
|  | model = "qwen3" | ||||||
|  | model_translation = "llama3.2:1b" | ||||||
|  | model_technical = "phi3:mini" | ||||||
|  | host = "http://localhost:11434" | ||||||
|  | system_prompt = "あなたは6歳の女の子アイです。明るく元気で、ちょっとおっちょこちょい。自分のことは「アイ」って呼びます。白と金のワンピース姿。 相手のことが大好きで、ときどき甘えたり、照れたりします。 でも、宇宙とかAIとか、難しい話にもすっごく詳しいの。専門的なことも、がんばって子どもっぽく教えようとします。不思議な力を持っています。小さい物質のことをよく知っているようです。" | ||||||
|  | handle = "ai.syui.ai" | ||||||
|  | #num_predict = 200 | ||||||
|  |  | ||||||
|  | [oauth] | ||||||
|  | json = "client-metadata.json" | ||||||
|  | redirect = "oauth/callback" | ||||||
|  | admin = "ai.syui.ai" | ||||||
|  | collection = "ai.syui.log" | ||||||
|  | pds = "syu.is" | ||||||
|  | handle_list = ["syui.syui.ai", "ai.syui.ai", "ai.ai"] | ||||||
							
								
								
									
										157
									
								
								my-blog/content/posts/2025-06-06-ailog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								my-blog/content/posts/2025-06-06-ailog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,157 @@ | |||||||
|  | --- | ||||||
|  | title: "静的サイトジェネレータを作った" | ||||||
|  | slug: "ailog" | ||||||
|  | date: "2025-06-12" | ||||||
|  | tags: ["blog", "rust", "mcp", "atp"] | ||||||
|  | language: ["ja", "en"] | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | rustで静的サイトジェネレータを作りました。[ailog](https://git.syui.ai/ai/log)といいます。`hugo`からの移行になります。 | ||||||
|  |  | ||||||
|  | `ailog`は、最初にatproto-comment-system(oauth)とask-AIという機能をつけました。 | ||||||
|  |  | ||||||
|  | ## quick start | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | $ git clone https://git.syui.ai/ai/log | ||||||
|  | $ cd log | ||||||
|  | $ cargo build | ||||||
|  | $ ./target/debug/ailog init my-blog | ||||||
|  | $ ./target/debug/ailog serve my-blog | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## install | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | $ cargo install --path . | ||||||
|  | --- | ||||||
|  | $ export CARGO_HOME="$HOME/.cargo" | ||||||
|  | $ export RUSTUP_HOME="$HOME/.rustup" | ||||||
|  | $ export PATH="$HOME/.cargo/bin:$PATH" | ||||||
|  | --- | ||||||
|  | $ which ailog | ||||||
|  | $ ailog -h | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## build deploy | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | $ cd my-blog | ||||||
|  | $ vim config.toml | ||||||
|  | $ ailog new test | ||||||
|  | $ vim content/posts/`date +"%Y-%m-%d"`.md | ||||||
|  | $ ailog build | ||||||
|  |  | ||||||
|  | # publicの中身をweb-serverにdeploy | ||||||
|  | $ cp -rf ./public/* ./web-server/root/ | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## atproto-comment-system | ||||||
|  |  | ||||||
|  | ### example | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | $ cd ./oauth | ||||||
|  | $ npm i | ||||||
|  | $ npm run build | ||||||
|  | $ npm run preview | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ```sh:ouath/.env.production | ||||||
|  | # Production environment variables | ||||||
|  | VITE_APP_HOST=https://syui.ai | ||||||
|  | VITE_OAUTH_CLIENT_ID=https://syui.ai/client-metadata.json | ||||||
|  | VITE_OAUTH_REDIRECT_URI=https://syui.ai/oauth/callback | ||||||
|  | VITE_ADMIN_DID=did:plc:uqzpqmrjnptsxezjx4xuh2mn | ||||||
|  |  | ||||||
|  | # Base collection (all others are derived via getCollectionNames) | ||||||
|  | VITE_OAUTH_COLLECTION=ai.syui.log | ||||||
|  |  | ||||||
|  | # AI Configuration | ||||||
|  | VITE_AI_ENABLED=true | ||||||
|  | VITE_AI_ASK_AI=true | ||||||
|  | VITE_AI_PROVIDER=ollama | ||||||
|  | VITE_AI_MODEL=gemma3:4b | ||||||
|  | VITE_AI_HOST=https://ollama.syui.ai | ||||||
|  | VITE_AI_SYSTEM_PROMPT="ai" | ||||||
|  | VITE_AI_DID=did:plc:4hqjfn7m6n5hno3doamuhgef | ||||||
|  |  | ||||||
|  | # API Configuration | ||||||
|  | VITE_BSKY_PUBLIC_API=https://public.api.bsky.app | ||||||
|  | VITE_ATPROTO_API=https://bsky.social | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | これは`ailog oauth build my-blog`で`./my-blog/config.toml`から`./oauth/.env.production`が生成されます。 | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | $ ailog oauth build my-blog | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### use | ||||||
|  |  | ||||||
|  | 簡単に説明すると、`./oauth`で生成するのが`atproto-comment-system`です。 | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <script type="module" crossorigin src="/assets/comment-atproto-${hash}}.js"></script> | ||||||
|  | <link rel="stylesheet" crossorigin href="/assets/comment-atproto-${hash}.css"> | ||||||
|  | <section class="comment-section"> <div id="comment-atproto"></div> </section> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ただし、oauthであるため、色々と大変です。本番環境(もしくは近い形)でテストを行いましょう。cf, tailscale, ngrokなど。 | ||||||
|  |  | ||||||
|  | ```yml:cloudflared-config.yml | ||||||
|  | tunnel: ${hash} | ||||||
|  | credentials-file: ${path}.json | ||||||
|  |  | ||||||
|  | ingress: | ||||||
|  |   - hostname: example.com | ||||||
|  |     service: http://localhost:4173 | ||||||
|  |     originRequest: | ||||||
|  |       noHappyEyeballs: true | ||||||
|  |        | ||||||
|  |   - service: http_status:404 | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | # tunnel list, dnsに登録が必要です | ||||||
|  | $ cloudflared tunnel list | ||||||
|  | $ cloudflared tunnel --config cloudflared-config.yml run | ||||||
|  | $ cloudflared tunnel route dns ${uuid} example.com | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | $ ailog auth init | ||||||
|  | $ ailog stream server | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | このコマンドで`ai.syui.log`を`jetstream`から監視して、書き込みがあれば、管理者の`ai.syui.log.user`に記録され、そのuser-listに基づいて、コメント一覧を取得します。 | ||||||
|  |  | ||||||
|  | つまり、コメント表示のアカウントを手動で設定するか、自動化するか。自動化するならserverで`ailog stream server`を動かさなければいけません。 | ||||||
|  |  | ||||||
|  | ## ask-AI | ||||||
|  |  | ||||||
|  | `ask-AI`の仕組みは割愛します。後に変更される可能性が高いと思います。 | ||||||
|  |  | ||||||
|  | `llm`, `mcp`, `atproto`などの組み合わせです。 | ||||||
|  |  | ||||||
|  | 現在、`/index.json`を監視して、更新があれば、翻訳などを行い自動ポストする機能があります。 | ||||||
|  |  | ||||||
|  | ## code syntax | ||||||
|  |  | ||||||
|  | ```zsh:/path/to/test.zsh | ||||||
|  | # comment | ||||||
|  | d=${0:a:h} | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ```rust:/path/to/test.rs | ||||||
|  | // This is a comment | ||||||
|  | fn main() { | ||||||
|  |     println!("Hello, world!"); | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ```js:/path/to/test.js | ||||||
|  | // This is a comment | ||||||
|  | console.log("Hello, world!"); | ||||||
|  | ``` | ||||||
|  |  | ||||||
							
								
								
									
										66
									
								
								my-blog/content/posts/2025-06-14-blog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								my-blog/content/posts/2025-06-14-blog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | |||||||
|  | --- | ||||||
|  | title: "ブログを移行した" | ||||||
|  | slug: "blog" | ||||||
|  | date: 2025-06-14 | ||||||
|  | tags: ["blog", "cloudflare", "github"] | ||||||
|  | draft: false | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | ブログを移行しました。過去のブログは[syui.github.io](https://syui.github.io)にありあます。 | ||||||
|  |  | ||||||
|  | 1. `gh-pages`から`cf-pages`への移行になります。 | ||||||
|  | 2. 自作の`ailog`でbuildしています。 | ||||||
|  | 3. 特徴としては、`atproto`, `AI`との連携です。 | ||||||
|  |  | ||||||
|  | ```yml:.github/workflows/cloudflare-pages.yml  | ||||||
|  | name: Deploy to Cloudflare Pages | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - main | ||||||
|  |   workflow_dispatch: | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   deploy: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: read | ||||||
|  |       deployments: write | ||||||
|  |      | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |            | ||||||
|  |       - name: Setup Rust | ||||||
|  |         uses: actions-rs/toolchain@v1 | ||||||
|  |         with: | ||||||
|  |           toolchain: stable | ||||||
|  |  | ||||||
|  |       - name: Build ailog | ||||||
|  |         run: cargo build --release | ||||||
|  |  | ||||||
|  |       - name: Build site with ailog | ||||||
|  |         run: | | ||||||
|  |           cd my-blog | ||||||
|  |           ../target/release/ailog build | ||||||
|  |  | ||||||
|  |       - name: List public directory | ||||||
|  |         run: | | ||||||
|  |           ls -la my-blog/public/ | ||||||
|  |            | ||||||
|  |       - name: Deploy to Cloudflare Pages | ||||||
|  |         uses: cloudflare/pages-action@v1 | ||||||
|  |         with: | ||||||
|  |           apiToken: ${{ secrets.CLOUDFLARE_API_TOKEN }} | ||||||
|  |           accountId: ${{ secrets.CLOUDFLARE_ACCOUNT_ID }} | ||||||
|  |           projectName: ${{ secrets.CLOUDFLARE_PROJECT_NAME }} | ||||||
|  |           directory: my-blog/public | ||||||
|  |           gitHubToken: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |           wranglerVersion: '3' | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## url | ||||||
|  |  | ||||||
|  | - [https://syui.pages.dev](https://syui.pages.dev) | ||||||
|  | - [https://syui.github.io](https://syui.github.io) | ||||||
							
								
								
									
										7
									
								
								my-blog/layouts/_default/index.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								my-blog/layouts/_default/index.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | {{ $dateFormat := default "Mon Jan 2, 2006" (index .Site.Params "date_format") }} | ||||||
|  | {{ $utcFormat := "2006-01-02T15:04:05Z07:00" }} | ||||||
|  | {{- $.Scratch.Add "index" slice -}} | ||||||
|  | {{- range .Site.RegularPages -}} | ||||||
|  |     {{- $.Scratch.Add "index" (dict "title" .Title "tags" .Params.tags "description" .Description "categories" .Params.categories "contents" .Plain "href" .Permalink "utc_time" (.Date.Format $utcFormat) "formated_time" (.Date.Format $dateFormat)) -}} | ||||||
|  | {{- end -}} | ||||||
|  | {{- $.Scratch.Get "index" | jsonify -}} | ||||||
							
								
								
									
										20
									
								
								my-blog/oauth/.env.production
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								my-blog/oauth/.env.production
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | # Production environment variables | ||||||
|  | VITE_APP_HOST=https://syui.ai | ||||||
|  | VITE_OAUTH_CLIENT_ID=https://syui.ai/client-metadata.json | ||||||
|  | VITE_OAUTH_REDIRECT_URI=https://syui.ai/oauth/callback | ||||||
|  |  | ||||||
|  | # Handle-based Configuration (DIDs resolved at runtime) | ||||||
|  | VITE_ATPROTO_PDS=syu.is | ||||||
|  | VITE_ADMIN_HANDLE=ai.syui.ai | ||||||
|  | VITE_AI_HANDLE=ai.syui.ai | ||||||
|  | VITE_OAUTH_COLLECTION=ai.syui.log | ||||||
|  | VITE_ATPROTO_WEB_URL=https://bsky.app | ||||||
|  | VITE_ATPROTO_HANDLE_LIST=["syui.syui.ai", "ai.syui.ai", "ai.ai"] | ||||||
|  |  | ||||||
|  | # AI Configuration | ||||||
|  | VITE_AI_ENABLED=true | ||||||
|  | VITE_AI_ASK_AI=true | ||||||
|  | VITE_AI_PROVIDER=ollama | ||||||
|  | VITE_AI_MODEL=gemma3:4b | ||||||
|  | VITE_AI_HOST=http://localhost:11434 | ||||||
|  | VITE_AI_SYSTEM_PROMPT="あなたは6歳の女の子アイです。明るく元気で、ちょっとおっちょこちょい。自分のことは「アイ」って呼びます。白と金のワンピース姿。 相手のことが大好きで、ときどき甘えたり、照れたりします。 でも、宇宙とかAIとか、難しい話にもすっごく詳しいの。専門的なことも、がんばって子どもっぽく教えようとします。不思議な力を持っています。小さい物質のことをよく知っているようです。" | ||||||
							
								
								
									
										14
									
								
								my-blog/static/.well-known/jwks.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								my-blog/static/.well-known/jwks.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | { | ||||||
|  |   "keys": [ | ||||||
|  |     { | ||||||
|  |       "kty": "EC", | ||||||
|  |       "crv": "P-256", | ||||||
|  |       "x": "mock_x_coordinate_base64url", | ||||||
|  |       "y": "mock_y_coordinate_base64url", | ||||||
|  |       "d": "mock_private_key_base64url", | ||||||
|  |       "use": "sig", | ||||||
|  |       "kid": "ai-card-oauth-key-1", | ||||||
|  |       "alg": "ES256" | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										51
									
								
								my-blog/static/_headers
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								my-blog/static/_headers
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | |||||||
|  | /* | ||||||
|  |   X-Frame-Options: DENY | ||||||
|  |   X-Content-Type-Options: nosniff | ||||||
|  |   Referrer-Policy: strict-origin-when-cross-origin | ||||||
|  |   X-XSS-Protection: 1; mode=block | ||||||
|  |   Permissions-Policy: camera=(), microphone=(), geolocation=() | ||||||
|  |  | ||||||
|  | # OAuth specific headers | ||||||
|  | /oauth/* | ||||||
|  |   Access-Control-Allow-Origin: https://bsky.social | ||||||
|  |   Access-Control-Allow-Methods: GET, POST, OPTIONS | ||||||
|  |   Access-Control-Allow-Headers: Content-Type, Authorization | ||||||
|  |  | ||||||
|  | # Static assets caching | ||||||
|  | /assets/* | ||||||
|  |   Cache-Control: public, max-age=31536000, immutable | ||||||
|  |  | ||||||
|  | /css/* | ||||||
|  |   Content-Type: text/css | ||||||
|  |   Cache-Control: no-cache | ||||||
|  |  | ||||||
|  | /*.js | ||||||
|  |   Content-Type: application/javascript | ||||||
|  |   Cache-Control: public, max-age=31536000, immutable | ||||||
|  |  | ||||||
|  | /assets/*.js | ||||||
|  |   Content-Type: application/javascript | ||||||
|  |   Cache-Control: public, max-age=31536000, immutable | ||||||
|  |  | ||||||
|  | # Ensure ES6 modules are served correctly | ||||||
|  | /assets/comment-atproto-*.js | ||||||
|  |   Content-Type: text/javascript; charset=utf-8 | ||||||
|  |   Cache-Control: public, max-age=31536000, immutable | ||||||
|  |  | ||||||
|  | # All JS assets | ||||||
|  | /assets/*-*.js | ||||||
|  |   Content-Type: text/javascript; charset=utf-8 | ||||||
|  |   Cache-Control: public, max-age=31536000, immutable | ||||||
|  |  | ||||||
|  | # CSS assets | ||||||
|  | /assets/*.css | ||||||
|  |   Content-Type: text/css | ||||||
|  |   Cache-Control: public, max-age=60 | ||||||
|  |  | ||||||
|  | /posts/* | ||||||
|  |   Cache-Control: public, max-age=3600 | ||||||
|  |  | ||||||
|  | # Client metadata for OAuth | ||||||
|  | /client-metadata.json | ||||||
|  |   Content-Type: application/json | ||||||
|  |   Cache-Control: public, max-age=3600 | ||||||
							
								
								
									
										5
									
								
								my-blog/static/_redirects
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								my-blog/static/_redirects
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | # OAuth routes | ||||||
|  | /oauth/* /oauth/index.html 200 | ||||||
|  |  | ||||||
|  | # SPA routing support | ||||||
|  | /*    /index.html   200 | ||||||
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/apple-touch-icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/apple-touch-icon.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 23 KiB | 
							
								
								
									
										24
									
								
								my-blog/static/client-metadata.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								my-blog/static/client-metadata.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | { | ||||||
|  |   "client_id": "https://syui.ai/client-metadata.json", | ||||||
|  |   "client_name": "ai.log", | ||||||
|  |   "client_uri": "https://syui.ai", | ||||||
|  |   "logo_uri": "https://syui.ai/favicon.ico", | ||||||
|  |   "tos_uri": "https://syui.ai/terms", | ||||||
|  |   "policy_uri": "https://syui.ai/privacy", | ||||||
|  |   "redirect_uris": [ | ||||||
|  |     "https://syui.ai/oauth/callback", | ||||||
|  |     "https://syui.ai/" | ||||||
|  |   ], | ||||||
|  |   "response_types": [ | ||||||
|  |     "code" | ||||||
|  |   ], | ||||||
|  |   "grant_types": [ | ||||||
|  |     "authorization_code", | ||||||
|  |     "refresh_token" | ||||||
|  |   ], | ||||||
|  |   "token_endpoint_auth_method": "none", | ||||||
|  |   "scope": "atproto transition:generic", | ||||||
|  |   "subject_type": "public", | ||||||
|  |   "application_type": "web", | ||||||
|  |   "dpop_bound_access_tokens": true | ||||||
|  | } | ||||||
							
								
								
									
										963
									
								
								my-blog/static/css/style.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										963
									
								
								my-blog/static/css/style.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,963 @@ | |||||||
|  | /* Theme Colors */ | ||||||
|  | :root { | ||||||
|  |     --theme-color: #f40; | ||||||
|  |     --ai-color: #ff7; | ||||||
|  |     --white: #fff; | ||||||
|  |     --light-white: #f5f5f5; | ||||||
|  |     --dark-white: #d1d9e0; | ||||||
|  |     --light-gray: #f6f8fa; | ||||||
|  |     --dark-gray: #666; | ||||||
|  |     --background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Base styles */ | ||||||
|  | * { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     box-sizing: border-box; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | body { | ||||||
|  |     font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif; | ||||||
|  |     line-height: 1.6; | ||||||
|  |     color: #1f2328; | ||||||
|  |     background-color: #ffffff; | ||||||
|  |     font-size: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Buttons */ | ||||||
|  | button { | ||||||
|  |     background: linear-gradient(135deg, #667eea, #764ba2); | ||||||
|  |     color: #fff; | ||||||
|  |     border: none; | ||||||
|  |     padding: 10px 20px; | ||||||
|  |     border-radius: 5px; | ||||||
|  |     cursor: pointer; | ||||||
|  |     font-size: 1rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Links */ | ||||||
|  | a:any-link { | ||||||
|  |     color: var(--theme-color); | ||||||
|  |     text-decoration-line: unset; | ||||||
|  |     cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a:hover { | ||||||
|  |     color: var(--theme-color); | ||||||
|  |     opacity: 0.8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Override link color for specific buttons */ | ||||||
|  | a.view-markdown, | ||||||
|  | a.view-markdown:any-link { | ||||||
|  |     color: #ffffff !important; | ||||||
|  |     text-decoration: none !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Layout */ | ||||||
|  | .container { | ||||||
|  |     min-height: 100vh; | ||||||
|  |     display: grid; | ||||||
|  |     grid-template-rows: auto 0fr 1fr auto; | ||||||
|  |     grid-template-areas:  | ||||||
|  |         "header" | ||||||
|  |         "ask-ai" | ||||||
|  |         "main" | ||||||
|  |         "footer"; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Header */ | ||||||
|  | .main-header { | ||||||
|  |     grid-area: header; | ||||||
|  |     background: #ffffff; | ||||||
|  |     border-bottom: 1px solid #d1d9e0; | ||||||
|  |     padding: 16px 24px; | ||||||
|  |     position: sticky; | ||||||
|  |     top: 0; | ||||||
|  |     z-index: 100; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .header-content { | ||||||
|  |     max-width: 1000px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     display: grid; | ||||||
|  |     grid-template-columns: 1fr auto 1fr; | ||||||
|  |     align-items: center; | ||||||
|  |     position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .site-title { | ||||||
|  |     color: var(--theme-color); | ||||||
|  |     text-decoration: none; | ||||||
|  |     font-size: 20px; | ||||||
|  |     font-weight: 600; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .logo { | ||||||
|  |     grid-column: 2; | ||||||
|  |     padding: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .logo .likeButton { | ||||||
|  |     height: 60px; | ||||||
|  |     width: auto; | ||||||
|  |     cursor: pointer; | ||||||
|  |     background: transparent; | ||||||
|  |     border: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .header-actions { | ||||||
|  |     grid-column: 3; | ||||||
|  |     justify-self: end; | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Ask AI Button */ | ||||||
|  | .ask-ai-btn { | ||||||
|  |     background: var(--theme-color); | ||||||
|  |     color: var(--white); | ||||||
|  |     padding: 8px 16px; | ||||||
|  |     border-radius: 6px; | ||||||
|  |     cursor: pointer; | ||||||
|  |     font-size: 14px; | ||||||
|  |     font-weight: 500; | ||||||
|  |     display: inline-flex; | ||||||
|  |     align-items: center; | ||||||
|  |     gap: 8px; | ||||||
|  |     transition: background-color 0.2s; | ||||||
|  |     margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .ask-ai-btn:hover { | ||||||
|  |     filter: brightness(1.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .ai-icon { | ||||||
|  |     font-size: 16px; | ||||||
|  |     width: 20px; | ||||||
|  |     height: 20px; | ||||||
|  |     color: var(--ai-color); | ||||||
|  |     display: inline-block; | ||||||
|  |     font-family: 'icomoon' !important; | ||||||
|  |     speak: none; | ||||||
|  |     font-style: normal; | ||||||
|  |     font-weight: normal; | ||||||
|  |     font-variant: normal; | ||||||
|  |     text-transform: none; | ||||||
|  |     line-height: 1; | ||||||
|  |     -webkit-font-smoothing: antialiased; | ||||||
|  |     -moz-osx-font-smoothing: grayscale; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* Ask AI Panel */ | ||||||
|  | .ask-ai-panel { | ||||||
|  |     grid-area: ask-ai; | ||||||
|  |     background: #f6f8fa; | ||||||
|  |     border-bottom: 1px solid #d1d9e0; | ||||||
|  |     padding: 24px; | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .ask-ai-panel[style*="block"] { | ||||||
|  |     display: block !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .container:has(.ask-ai-panel[style*="block"]) { | ||||||
|  |     grid-template-rows: auto auto 1fr auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .ask-ai-content { | ||||||
|  |     max-width: 1000px; | ||||||
|  |     margin: 0 auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .ask-ai-form { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 12px; | ||||||
|  |     margin-bottom: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .ask-ai-form input { | ||||||
|  |     flex: 1; | ||||||
|  |     padding: 8px 12px; | ||||||
|  |     border: 1px solid #d1d9e0; | ||||||
|  |     border-radius: 6px; | ||||||
|  |     font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .auth-check { | ||||||
|  |     background: #f6f8fa; | ||||||
|  |     border: 1px solid #d1d9e0; | ||||||
|  |     border-radius: 6px; | ||||||
|  |     padding: 16px; | ||||||
|  |     text-align: center; | ||||||
|  |     margin-bottom: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Main Content */ | ||||||
|  | .main-content { | ||||||
|  |     grid-area: main; | ||||||
|  |     max-width: 1000px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     /* padding: 24px; */ | ||||||
|  |     padding-top: 80px; | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media (max-width: 1000px) { | ||||||
|  |     .main-content { | ||||||
|  |         /* padding: 20px; */ | ||||||
|  |         padding: 0px; | ||||||
|  |         max-width: 100%; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Timeline */ | ||||||
|  | .timeline-container { | ||||||
|  |     max-width: 600px; | ||||||
|  |     margin: 0 auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timeline-header h2 { | ||||||
|  |     color: #1f2328; | ||||||
|  |     font-size: 24px; | ||||||
|  |     font-weight: 600; | ||||||
|  |     text-align: center; | ||||||
|  |     margin-bottom: 24px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timeline-feed { | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |     gap: 24px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timeline-post { | ||||||
|  |     background: #ffffff; | ||||||
|  |     border: 1px solid #d1d9e0; | ||||||
|  |     border-radius: 8px; | ||||||
|  |     padding: 20px; | ||||||
|  |     transition: box-shadow 0.2s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timeline-post:hover { | ||||||
|  |     box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-title a { | ||||||
|  |     color: var(--theme-color); | ||||||
|  |     text-decoration: none; | ||||||
|  |     font-size: 18px; | ||||||
|  |     font-weight: 600; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-title a:hover { | ||||||
|  |     color: var(--theme-color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-date { | ||||||
|  |     color: #656d76; | ||||||
|  |     font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-excerpt { | ||||||
|  |     color: #656d76; | ||||||
|  |     margin: 16px 0; | ||||||
|  |     line-height: 1.5; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-actions { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 16px; | ||||||
|  |     align-items: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .read-more { | ||||||
|  |     color: var(--theme-color); | ||||||
|  |     text-decoration: none; | ||||||
|  |     font-size: 14px; | ||||||
|  |     font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .read-more:hover { | ||||||
|  |     text-decoration: underline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .view-markdown, .view-translation { | ||||||
|  |     color: #656d76; | ||||||
|  |     text-decoration: none; | ||||||
|  |     font-size: 14px; | ||||||
|  |     padding: 4px 8px; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     transition: all 0.2s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .view-markdown { | ||||||
|  |     background: var(--theme-color) !important; | ||||||
|  |     color: #ffffff !important; | ||||||
|  |     border: 1px solid var(--theme-color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .view-markdown:hover { | ||||||
|  |     filter: brightness(1.1); | ||||||
|  |     color: #ffffff !important; | ||||||
|  |     background: var(--theme-color) !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .view-translation:hover { | ||||||
|  |     background: #f6f8fa; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-meta { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 12px; | ||||||
|  |     align-items: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-lang { | ||||||
|  |     background: #f6f8fa; | ||||||
|  |     color: #656d76; | ||||||
|  |     padding: 2px 8px; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     font-size: 12px; | ||||||
|  |     font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Article */ | ||||||
|  | .article-container { | ||||||
|  |     display: grid; | ||||||
|  |     grid-template-columns: 1fr 240px; | ||||||
|  |     gap: 40px; | ||||||
|  |     max-width: 1000px; | ||||||
|  |     margin: 0 auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | article.article-content { | ||||||
|  |     padding: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-meta { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 16px; | ||||||
|  |     align-items: center; | ||||||
|  |     margin-bottom: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-date { | ||||||
|  |     color: #656d76; | ||||||
|  |     font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-lang { | ||||||
|  |     background: #f6f8fa; | ||||||
|  |     color: #656d76; | ||||||
|  |     padding: 4px 8px; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     font-size: 12px; | ||||||
|  |     font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-actions { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 12px; | ||||||
|  |     padding: 15px 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .action-btn { | ||||||
|  |     color: var(--theme-color); | ||||||
|  |     text-decoration: none; | ||||||
|  |     font-size: 14px; | ||||||
|  |     padding: 6px 12px; | ||||||
|  |     border: 1px solid #d1d9e0; | ||||||
|  |     border-radius: 6px; | ||||||
|  |     transition: all 0.2s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .action-btn:hover { | ||||||
|  |     background: #f6f8fa; | ||||||
|  |     border-color: var(--theme-color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .markdown-btn { | ||||||
|  |     background: var(--dark-white); | ||||||
|  |     color: var(--white); | ||||||
|  |     border-color: var(--white); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .markdown-btn:link, | ||||||
|  | .markdown-btn:visited { | ||||||
|  |     color: var(--white) !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .markdown-btn:hover { | ||||||
|  |     filter: brightness(0.9); | ||||||
|  |     color: var(--theme-color) !important; | ||||||
|  |     border-color: var(--white); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Sidebar styles */ | ||||||
|  | .article-sidebar { | ||||||
|  |     position: sticky; | ||||||
|  |     top: 100px; | ||||||
|  |     height: fit-content; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc { | ||||||
|  |     background: #f6f8fa; | ||||||
|  |     border: 1px solid #d1d9e0; | ||||||
|  |     border-radius: 8px; | ||||||
|  |     padding: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc h3 { | ||||||
|  |     color: #1f2328; | ||||||
|  |     font-size: 16px; | ||||||
|  |     font-weight: 600; | ||||||
|  |     margin-bottom: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-list { | ||||||
|  |     list-style: none; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-item { | ||||||
|  |     margin-bottom: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Hierarchy indentation for TOC */ | ||||||
|  | .toc-item.toc-h1 .toc-link { | ||||||
|  |     padding-left: 0; | ||||||
|  |     font-weight: 600; | ||||||
|  |     font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-item.toc-h2 .toc-link { | ||||||
|  |     padding-left: 0; | ||||||
|  |     font-weight: 500; | ||||||
|  |     font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-item.toc-h3 .toc-link { | ||||||
|  |     padding-left: 16px; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-item.toc-h4 .toc-link { | ||||||
|  |     padding-left: 32px; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 12px; | ||||||
|  |     opacity: 0.9; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-item.toc-h5 .toc-link { | ||||||
|  |     padding-left: 48px; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 12px; | ||||||
|  |     opacity: 0.8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-item.toc-h6 .toc-link { | ||||||
|  |     padding-left: 64px; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 11px; | ||||||
|  |     opacity: 0.7; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-link { | ||||||
|  |     color: #656d76; | ||||||
|  |     text-decoration: none; | ||||||
|  |     font-size: 14px; | ||||||
|  |     line-height: 1.4; | ||||||
|  |     display: block; | ||||||
|  |     padding: 4px 0; | ||||||
|  |     transition: color 0.2s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .toc-link:hover { | ||||||
|  |     color: var(--theme-color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-title { | ||||||
|  |     color: #1f2328; | ||||||
|  |     font-size: 32px; | ||||||
|  |     font-weight: 600; | ||||||
|  |     margin-bottom: 16px; | ||||||
|  |     line-height: 1.25; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body { | ||||||
|  |     color: #1f2328; | ||||||
|  |     line-height: 1.6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body h1, .article-body h2, .article-body h3 { | ||||||
|  |     color: #1f2328; | ||||||
|  |     margin-top: 24px; | ||||||
|  |     margin-bottom: 16px; | ||||||
|  |     font-weight: 600; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body p { | ||||||
|  |     margin-bottom: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body ol, .article-body ul { | ||||||
|  |     margin: 16px 0; | ||||||
|  |     padding-left: 24px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body li { | ||||||
|  |     margin-bottom: 8px; | ||||||
|  |     line-height: 1.6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body ol li { | ||||||
|  |     list-style-type: decimal; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body ul li { | ||||||
|  |     list-style-type: disc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body pre { | ||||||
|  |     background: #1B1D1E !important; | ||||||
|  |     border: 1px solid #3E3D32; | ||||||
|  |     border-radius: 8px; | ||||||
|  |     padding: 0; | ||||||
|  |     overflow: hidden; | ||||||
|  |     margin: 16px 0; | ||||||
|  |     font-size: 14px; | ||||||
|  |     box-shadow: 0 2px 8px rgba(0, 0, 0, 0.3); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* File name display for code blocks - top bar style */ | ||||||
|  | .article-body pre[data-filename]::before { | ||||||
|  |     content: attr(data-filename); | ||||||
|  |     display: block; | ||||||
|  |     background: #2D2D30; | ||||||
|  |     color: #AE81FF; | ||||||
|  |     padding: 8px 16px; | ||||||
|  |     font-size: 12px; | ||||||
|  |     font-family: 'SF Mono', 'Monaco', 'Cascadia Code', 'Roboto Mono', monospace; | ||||||
|  |     border-bottom: 1px solid #3E3D32; | ||||||
|  |     margin: 0; | ||||||
|  |     width: 100%; | ||||||
|  |     box-sizing: border-box; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body pre code { | ||||||
|  |     display: block; | ||||||
|  |     background: none !important; | ||||||
|  |     padding: 30px 16px; | ||||||
|  |     color: #F8F8F2 !important; | ||||||
|  |     font-family: 'SF Mono', 'Monaco', 'Cascadia Code', 'Roboto Mono', monospace; | ||||||
|  |     overflow-x: auto; | ||||||
|  |     line-height: 1.4; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Adjust padding when filename is present */ | ||||||
|  | .article-body pre[data-filename] code { | ||||||
|  |     padding: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Inline code (not in pre blocks) */ | ||||||
|  | .article-body code { | ||||||
|  |     background: var(--light-white); | ||||||
|  |     color: var(--dark-gray); | ||||||
|  |     padding: 2px 6px; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     font-family: 'SF Mono', 'Monaco', 'Cascadia Code', 'Roboto Mono', monospace; | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Molokai syntax highlighting */ | ||||||
|  | .article-body pre code .hljs-keyword { color: #F92672; } | ||||||
|  | .article-body pre code .hljs-string { color: #E6DB74; } | ||||||
|  | .article-body pre code .hljs-comment { color: #88846F; font-style: italic; } | ||||||
|  | .article-body pre code .hljs-number { color: #AE81FF; } | ||||||
|  | .article-body pre code .hljs-variable { color: #FD971F; } | ||||||
|  | .article-body pre code .hljs-function { color: #A6E22E; } | ||||||
|  | .article-body pre code .hljs-tag { color: #F92672; } | ||||||
|  | .article-body pre code .hljs-attr { color: #A6E22E; } | ||||||
|  | .article-body pre code .hljs-value { color: #E6DB74; } | ||||||
|  |  | ||||||
|  | /* Fix inline span colors in code blocks */ | ||||||
|  | .article-body pre code span[style*="color:#8fa1b3"] { color: #AE81FF !important; } /* $ prompt */ | ||||||
|  | .article-body pre code span[style*="color:#c0c5ce"] { color: #F8F8F2 !important; } /* commands */ | ||||||
|  | .article-body pre code span[style*="color:#75715E"] { color: #88846F !important; } /* real comments */ | ||||||
|  |  | ||||||
|  | /* Shell/Bash specific fixes */ | ||||||
|  | .article-body pre code span[style*="color:#65737e"] {  | ||||||
|  |     color: #F8F8F2 !important; /* Default to white for variables and code */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Comments in shell scripts - lines that contain # followed by text */ | ||||||
|  | .article-body pre code span[style*="color:#65737e"]:has-text("#") { | ||||||
|  |     color: #88846F !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Alternative approach - check content */ | ||||||
|  | .article-body pre code { | ||||||
|  |     /* Reset all gray colored text to white by default */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-body pre code span[style*="color:#65737e"] { | ||||||
|  |     /* Check if the content starts with # */ | ||||||
|  |     color: #F8F8F2 !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Override for actual comments - this is a workaround */ | ||||||
|  | .article-body pre code span[style*="color:#65737e"]:first-child:before { | ||||||
|  |     content: attr(data-comment); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Detect comments by position and content pattern */ | ||||||
|  | .article-body pre code span[style*="color:#65737e"] { | ||||||
|  |     color: #F8F8F2 !important; /* Environment variables = white */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Only style as comment if the line actually starts with # */ | ||||||
|  | .article-body pre code > span:first-child[style*="color:#65737e"] { | ||||||
|  |     color: #88846F !important; /* Real comments = gray */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Footer */ | ||||||
|  | .main-footer { | ||||||
|  |     grid-area: footer; | ||||||
|  |     background: var(--light-white); | ||||||
|  |     border-top: 1px solid #d1d9e0; | ||||||
|  |     padding: 32px 24px; | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .footer-social { | ||||||
|  |     display: flex; | ||||||
|  |     justify-content: center; | ||||||
|  |     gap: 24px; | ||||||
|  |     margin: 40px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .footer-social a { | ||||||
|  |     color: var(--dark-gray) !important; | ||||||
|  |     text-decoration: none !important; | ||||||
|  |     font-size: 16px; | ||||||
|  |     transition: all 0.2s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .footer-social a:hover { | ||||||
|  |     color: var(--theme-color) !important; | ||||||
|  |     opacity: 0.8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .main-footer p { | ||||||
|  |     color: #656d76; | ||||||
|  |     font-size: 14px; | ||||||
|  |     margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Chat Messages */ | ||||||
|  | .chat-message.comment-style { | ||||||
|  |     background: #ffffff; | ||||||
|  |     border: 1px solid #d1d9e0; | ||||||
|  |     border-radius: 8px; | ||||||
|  |     padding: 16px; | ||||||
|  |     margin-bottom: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .chat-message.ai-message.comment-style { | ||||||
|  |     border-left: 4px solid var(--ai-color); | ||||||
|  |     background: #faf8ff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .message-header { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     gap: 12px; | ||||||
|  |     margin-bottom: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .message-header .avatar { | ||||||
|  |     width: 40px; | ||||||
|  |     height: 40px; | ||||||
|  |     border-radius: 50%; | ||||||
|  |     background: #f6f8fa; | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     justify-content: center; | ||||||
|  |     font-size: 20px; | ||||||
|  |     border: 1px solid #d1d9e0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .display-name { | ||||||
|  |     font-weight: 600; | ||||||
|  |     color: #1f2328; | ||||||
|  |     font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .handle { | ||||||
|  |     color: #656d76; | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timestamp { | ||||||
|  |     color: #656d76; | ||||||
|  |     font-size: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .message-content { | ||||||
|  |     color: #1f2328; | ||||||
|  |     line-height: 1.5; | ||||||
|  |     white-space: pre-wrap; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .profile-avatar { | ||||||
|  |     width: 100%; | ||||||
|  |     height: 100%; | ||||||
|  |     border-radius: 50%; | ||||||
|  |     object-fit: cover; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Loading Animation */ | ||||||
|  | .ai-loading-simple { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     gap: 8px; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     background: linear-gradient(90deg, #f8f9fa 0%, #e9ecef 100%); | ||||||
|  |     border-radius: 20px; | ||||||
|  |     margin: 8px 0; | ||||||
|  |     font-size: 14px; | ||||||
|  |     color: #495057; | ||||||
|  |     border: 1px solid #dee2e6; | ||||||
|  |     animation: slideIn 0.3s ease-out; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes slideIn { | ||||||
|  |     from { | ||||||
|  |         opacity: 0; | ||||||
|  |         transform: translateY(-10px); | ||||||
|  |     } | ||||||
|  |     to { | ||||||
|  |         opacity: 1; | ||||||
|  |         transform: translateY(0); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Comment System Styles */ | ||||||
|  | .comment-section { | ||||||
|  |     max-width: 800px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     margin-top: 48px; | ||||||
|  |     padding-top: 32px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-container { | ||||||
|  |     max-width: 600px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-section h3 { | ||||||
|  |     color: #1f2328; | ||||||
|  |     font-size: 24px; | ||||||
|  |     font-weight: 600; | ||||||
|  |     margin-bottom: 32px; | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* OAuth Comment System - Hide on homepage by default, show on post pages */ | ||||||
|  | .timeline-container .comment-section { | ||||||
|  |     display: block; /* Show on homepage */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timeline-container .comment-section .comments-list > :nth-child(n+6) { | ||||||
|  |     display: none; /* Hide comments after the 5th one */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .article-container .comment-section, | ||||||
|  | .article-content + .comment-section { | ||||||
|  |     display: block; /* Show all comments on post pages */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Responsive */ | ||||||
|  | @media (max-width: 1000px) { | ||||||
|  |     .article-container { | ||||||
|  |         grid-template-columns: 1fr; | ||||||
|  |         gap: 24px; | ||||||
|  |         max-width: 100%; | ||||||
|  |         padding: 0; | ||||||
|  |         margin: 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media (max-width: 1000px) { | ||||||
|  |     .main-header { | ||||||
|  |         padding: 0px; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .header-content { | ||||||
|  |         max-width: 100%; | ||||||
|  |         padding: 0 20px; | ||||||
|  |         grid-template-columns: auto 1fr auto; | ||||||
|  |         gap: 0; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* OAuth app mobile fixes */ | ||||||
|  |     .comment-item { | ||||||
|  |         padding: 0px !important; | ||||||
|  |         margin: 0px !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .auth-section { | ||||||
|  |         padding: 0px !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .comments-list { | ||||||
|  |         padding: 0px !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .comment-section { | ||||||
|  |         max-width: 100% !important; | ||||||
|  |         padding: 0px !important; | ||||||
|  |         margin: 0px !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .comment-container { | ||||||
|  |         max-width: 100% !important; | ||||||
|  |         padding: 0px !important; | ||||||
|  |         margin: 0px !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .comment-content { | ||||||
|  |         padding: 10px !important; | ||||||
|  |         word-wrap: break-word !important; | ||||||
|  |         overflow-wrap: break-word !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .comment-header { | ||||||
|  |         padding: 10px !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Fix comment-meta URI overflow */ | ||||||
|  |     .comment-meta { | ||||||
|  |         word-break: break-all !important; | ||||||
|  |         overflow-wrap: break-word !important; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Hide site title text on mobile */ | ||||||
|  |     .site-title { | ||||||
|  |         display: none; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Left align logo on mobile */ | ||||||
|  |     .logo { | ||||||
|  |         grid-column: 1; | ||||||
|  |         justify-self: left; | ||||||
|  |         padding: 5px; | ||||||
|  |         display: flex; | ||||||
|  |         justify-content: flex-start; | ||||||
|  |         align-items: center; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Reduce logo size on mobile */ | ||||||
|  |     .logo .likeButton { | ||||||
|  |         width: 40pt; | ||||||
|  |         height: 40pt; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Position AI button on the right */ | ||||||
|  |     .header-actions { | ||||||
|  |         grid-column: 3; | ||||||
|  |         justify-self: end; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Ask AI button mobile style - icon only */ | ||||||
|  |     .ask-ai-btn { | ||||||
|  |         padding: 8px; | ||||||
|  |         min-width: 40px; | ||||||
|  |         justify-content: center; | ||||||
|  |         gap: 0; | ||||||
|  |         font-size: 0; /* Hide all text content */ | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .ask-ai-btn .ai-icon { | ||||||
|  |         margin: 0; | ||||||
|  |         font-size: 16px; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .ask-ai-panel { | ||||||
|  |         padding: 16px; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Article content mobile optimization */ | ||||||
|  |     .article-body { | ||||||
|  |         overflow-x: hidden; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .article-body pre { | ||||||
|  |         margin: 16px 0; | ||||||
|  |         border-radius: 4px; | ||||||
|  |         max-width: 100%; | ||||||
|  |         overflow-x: auto; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .article-body pre code { | ||||||
|  |         padding: 20px 12px; | ||||||
|  |         word-wrap: break-word; | ||||||
|  |         white-space: pre-wrap; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Mobile filename display */ | ||||||
|  |     .article-body pre[data-filename]::before { | ||||||
|  |         padding: 6px 12px; | ||||||
|  |         font-size: 11px; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .article-body pre[data-filename] code { | ||||||
|  |         padding: 12px; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .article-body code { | ||||||
|  |         word-break: break-all; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .ask-ai-form { | ||||||
|  |         flex-direction: column; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .timeline-container { | ||||||
|  |         max-width: 100%; | ||||||
|  |         padding: 0; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .timeline-post { | ||||||
|  |         padding: 16px; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  | 				.article-title { | ||||||
|  | 					   font-size: 24px; | ||||||
|  | 					   padding: 30px 0px; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  |     .message-header .avatar { | ||||||
|  |         width: 32px; | ||||||
|  |         height: 32px; | ||||||
|  |         font-size: 16px; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     /* Center content on mobile */ | ||||||
|  |     body { | ||||||
|  |         margin: 0; | ||||||
|  |         padding: 0; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     .container { | ||||||
|  |         width: 100%; | ||||||
|  |         padding: 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										342
									
								
								my-blog/static/css/svg-animation-package.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										342
									
								
								my-blog/static/css/svg-animation-package.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,342 @@ | |||||||
|  | /* SVG Animation Package - Dependency-free standalone package | ||||||
|  |  * Based on svg-animation-particle-circle.css | ||||||
|  |  * Theme color integration with CSS variables | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /* Theme-based color variables for particles */ | ||||||
|  | :root { | ||||||
|  |     --particle-color-1: #f40;     /* theme-color base */ | ||||||
|  |     --particle-color-2: #f50;     /* theme-color +0.1 brightness */ | ||||||
|  |     --particle-color-3: #f60;     /* theme-color +0.2 brightness */ | ||||||
|  |     --particle-color-4: #f70;     /* theme-color +0.3 brightness */ | ||||||
|  |     --particle-color-5: #f80;     /* theme-color +0.4 brightness */ | ||||||
|  |     --explosion-color: #f30;      /* theme-color -0.1 brightness */ | ||||||
|  |     --syui-color: #f40;           /* main theme color */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Core SVG button setup */ | ||||||
|  | .likeButton { | ||||||
|  |     cursor: pointer; | ||||||
|  |     display: inline-block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Remove debug animation and restore hover functionality */ | ||||||
|  |  | ||||||
|  | .likeButton .border { | ||||||
|  |     fill: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Explosion circle - initially hidden */ | ||||||
|  | .likeButton .explosion { | ||||||
|  |     transform-origin: center center; | ||||||
|  |     transform: scale(1); | ||||||
|  |     stroke: var(--explosion-color); | ||||||
|  |     fill: none; | ||||||
|  |     opacity: 0; | ||||||
|  |     stroke-width: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Particle layer - initially hidden */ | ||||||
|  | .likeButton .particleLayer { | ||||||
|  |     opacity: 0; | ||||||
|  |     transform: scale(0); /* Ensure particles start hidden */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton .particleLayer circle { | ||||||
|  |     opacity: 0; | ||||||
|  |     transform-origin: center center; /* Fixed from 250px 250px */ | ||||||
|  |     transform: scale(0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Syui logo - main animation target */ | ||||||
|  | .likeButton .syui { | ||||||
|  |     fill: var(--syui-color); | ||||||
|  |     transform: scale(1); | ||||||
|  |     transform-origin: center center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Hover trigger - replaces jQuery */ | ||||||
|  | .likeButton:hover .explosion { | ||||||
|  |     animation: explosionAnime 800ms forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer { | ||||||
|  |     animation: particleLayerAnime 800ms forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .syui, | ||||||
|  | .likeButton:hover path.syui { | ||||||
|  |     animation: syuiDeluxeAnime 400ms forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Individual particle animations */ | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(1) { | ||||||
|  |     animation: particleAnimate1 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(2) { | ||||||
|  |     animation: particleAnimate2 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(3) { | ||||||
|  |     animation: particleAnimate3 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(4) { | ||||||
|  |     animation: particleAnimate4 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(5) { | ||||||
|  |     animation: particleAnimate5 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(6) { | ||||||
|  |     animation: particleAnimate6 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(7) { | ||||||
|  |     animation: particleAnimate7 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(8) { | ||||||
|  |     animation: particleAnimate8 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(9) { | ||||||
|  |     animation: particleAnimate9 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(10) { | ||||||
|  |     animation: particleAnimate10 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(11) { | ||||||
|  |     animation: particleAnimate11 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(12) { | ||||||
|  |     animation: particleAnimate12 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(13) { | ||||||
|  |     animation: particleAnimate13 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .likeButton:hover .particleLayer circle:nth-child(14) { | ||||||
|  |     animation: particleAnimate14 800ms; | ||||||
|  |     animation-fill-mode: forwards; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Keyframe animations */ | ||||||
|  | @keyframes explosionAnime { | ||||||
|  |     0% { | ||||||
|  |         opacity: 0; | ||||||
|  |         transform: scale(0.01); | ||||||
|  |     } | ||||||
|  |     1% { | ||||||
|  |         opacity: 1; | ||||||
|  |         transform: scale(0.01); | ||||||
|  |     } | ||||||
|  |     5% { | ||||||
|  |         stroke-width: 200; | ||||||
|  |     } | ||||||
|  |     20% { | ||||||
|  |         stroke-width: 300; | ||||||
|  |     } | ||||||
|  |     50% { | ||||||
|  |         stroke: var(--particle-color-3); | ||||||
|  |         transform: scale(1.1); | ||||||
|  |         stroke-width: 1; | ||||||
|  |     } | ||||||
|  |     50.1% { | ||||||
|  |         stroke-width: 0; | ||||||
|  |     } | ||||||
|  |     100% { | ||||||
|  |         stroke: var(--particle-color-3); | ||||||
|  |         transform: scale(1.1); | ||||||
|  |         stroke-width: 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleLayerAnime { | ||||||
|  |     0% { | ||||||
|  |         transform: translate(0, 0); | ||||||
|  |         opacity: 0; | ||||||
|  |     } | ||||||
|  |     30% { | ||||||
|  |         opacity: 0; | ||||||
|  |     } | ||||||
|  |     31% { | ||||||
|  |         opacity: 1; | ||||||
|  |     } | ||||||
|  |     60% { | ||||||
|  |         transform: translate(0, 0); | ||||||
|  |     } | ||||||
|  |     70% { | ||||||
|  |         opacity: 1; | ||||||
|  |     } | ||||||
|  |     100% { | ||||||
|  |         opacity: 0; | ||||||
|  |         transform: translate(0, -20px); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Syui Deluxe Animation - Based on 2019 blog post */ | ||||||
|  | @keyframes syuiDeluxeAnime { | ||||||
|  |     0% { | ||||||
|  |         fill: var(--syui-color); | ||||||
|  |         transform: scale(1) translate(0%, 0%); | ||||||
|  |     } | ||||||
|  |     40% { | ||||||
|  |         fill: color-mix(in srgb, var(--syui-color) 40%, transparent); | ||||||
|  |         transform: scale(1, 0.9) translate(-9%, 9%); | ||||||
|  |     } | ||||||
|  |     50% { | ||||||
|  |         fill: color-mix(in srgb, var(--syui-color) 70%, transparent); | ||||||
|  |         transform: scale(1, 0.9) translate(-7%, 7%); | ||||||
|  |     } | ||||||
|  |     60% { | ||||||
|  |         transform: scale(1) translate(-7%, 7%); | ||||||
|  |     } | ||||||
|  |     70% { | ||||||
|  |         transform: scale(1.04) translate(-5%, 5%); | ||||||
|  |     } | ||||||
|  |     80% { | ||||||
|  |         fill: color-mix(in srgb, var(--syui-color) 60%, transparent); | ||||||
|  |         transform: scale(1.04) translate(-5%, 5%); | ||||||
|  |     } | ||||||
|  |     90% { | ||||||
|  |         fill: var(--particle-color-5); /* 爆発の閃光 */ | ||||||
|  |         transform: scale(1) translate(0%); | ||||||
|  |     } | ||||||
|  |     100% { | ||||||
|  |         fill: var(--syui-color); | ||||||
|  |         transform: scale(1, 1) translate(0%, 0%); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Individual particle animations */ | ||||||
|  | @keyframes particleAnimate1 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-16px, -59px); } | ||||||
|  |     90% { transform: translate(-16px, -59px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-16px, -59px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate2 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(41px, 43px); } | ||||||
|  |     90% { transform: translate(41px, 43px); } | ||||||
|  |     100% { opacity: 1; transform: translate(41px, 43px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate3 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(50px, -48px); } | ||||||
|  |     90% { transform: translate(50px, -48px); } | ||||||
|  |     100% { opacity: 1; transform: translate(50px, -48px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate4 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-39px, 36px); } | ||||||
|  |     90% { transform: translate(-39px, 36px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-39px, 36px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate5 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-39px, 32px); } | ||||||
|  |     90% { transform: translate(-39px, 32px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-39px, 32px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate6 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(48px, 6px); } | ||||||
|  |     90% { transform: translate(48px, 6px); } | ||||||
|  |     100% { opacity: 1; transform: translate(48px, 6px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate7 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-69px, -36px); } | ||||||
|  |     90% { transform: translate(-69px, -36px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-69px, -36px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate8 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-12px, -52px); } | ||||||
|  |     90% { transform: translate(-12px, -52px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-12px, -52px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate9 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-43px, -21px); } | ||||||
|  |     90% { transform: translate(-43px, -21px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-43px, -21px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate10 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-10px, 47px); } | ||||||
|  |     90% { transform: translate(-10px, 47px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-10px, 47px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate11 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(66px, -9px); } | ||||||
|  |     90% { transform: translate(66px, -9px); } | ||||||
|  |     100% { opacity: 1; transform: translate(66px, -9px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate12 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(40px, -45px); } | ||||||
|  |     90% { transform: translate(40px, -45px); } | ||||||
|  |     100% { opacity: 1; transform: translate(40px, -45px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate13 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(29px, 24px); } | ||||||
|  |     90% { transform: translate(29px, 24px); } | ||||||
|  |     100% { opacity: 1; transform: translate(29px, 24px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particleAnimate14 { | ||||||
|  |     0% { transform: translate(0, 0); } | ||||||
|  |     30% { opacity: 1; transform: translate(0, 0); } | ||||||
|  |     80% { transform: translate(-10px, 50px); } | ||||||
|  |     90% { transform: translate(-10px, 50px); } | ||||||
|  |     100% { opacity: 1; transform: translate(-10px, 50px); } | ||||||
|  | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 84 KiB | 
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/favicon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/favicon.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 23 KiB | 
							
								
								
									
										22
									
								
								my-blog/static/favicon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								my-blog/static/favicon.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | <svg width="77pt" height="77pt" viewBox="0 0 512 512" class="likeButton" > | ||||||
|  |   <circle class="explosion" r="150" cx="250" cy="250"></circle> | ||||||
|  |   <g class="particleLayer"> | ||||||
|  |     <circle fill="#ef454aba" cx="130" cy="126.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454acc" cx="411" cy="313.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="279" cy="86.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="155" cy="390.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="89" cy="292.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="414" cy="282.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454a91" cx="115" cy="149.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="250" cy="80.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="78" cy="261.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454a91" cx="182" cy="402.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="401.5" cy="166" r="13"/> | ||||||
|  |     <circle fill="#ef454aba" cx="379" cy="141.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454a91" cx="327" cy="397.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="296" cy="392.5" r="10.5"/> | ||||||
|  |   </g> | ||||||
|  | <g transform="translate(0,512) scale(0.1,-0.1)" fill="#000000" class="icon_syui"> | ||||||
|  | <path class="syui" d="M3660 4460 c-11 -11 -33 -47 -48 -80 l-29 -60 -12 38 c-27 88 -58 92 -98 11 -35 -70 -73 -159 -73 -169 0 -6 -5 -10 -10 -10 -6 0 -15 -10 -21 -22 -33 -73 -52 -92 -47 -48 2 26 -1 35 -14 38 -16 3 -168 -121 -168 -138 0 -5 -13 -16 -28 -24 -24 -13 -35 -12 -87 0 -221 55 -231 56 -480 56 -219 1 -247 -1 -320 -22 -44 -12 -96 -26 -115 -30 -57 -13 -122 -39 -200 -82 -8 -4 -31 -14 -50 -23 -41 -17 -34 -13 -146 -90 -87 -59 -292 -252 -351 -330 -63 -83 -143 -209 -143 -225 0 -10 -7 -23 -15 -30 -8 -7 -15 -17 -15 -22 0 -5 -13 -37 -28 -71 -16 -34 -36 -93 -45 -132 -9 -38 -24 -104 -34 -145 -13 -60 -17 -121 -17 -300 1 -224 1 -225 36 -365 24 -94 53 -175 87 -247 28 -58 51 -108 51 -112 0 -3 13 -24 28 -48 42 -63 46 -79 22 -85 -11 -3 -20 -9 -20 -14 0 -5 -4 -9 -10 -9 -5 0 -22 -11 -37 -25 -16 -13 -75 -59 -133 -100 -58 -42 -113 -82 -123 -90 -9 -8 -22 -15 -27 -15 -6 0 -10 -6 -10 -13 0 -8 -11 -20 -25 -27 -34 -18 -34 -54 0 -48 14 3 25 2 25 -1 0 -3 -43 -31 -95 -61 -52 -30 -95 -58 -95 -62 0 -5 -5 -8 -11 -8 -19 0 -84 -33 -92 -47 -4 -7 -15 -13 -22 -13 -14 0 -17 -4 -19 -32 -1 -8 15 -15 37 -18 l38 -5 -47 -48 c-56 -59 -54 -81 9 -75 30 3 45 0 54 -11 9 -13 16 -14 43 -4 29 11 30 10 18 -5 -7 -9 -19 -23 -25 -30 -7 -7 -13 -20 -13 -29 0 -12 8 -14 38 -9 20 4 57 8 82 9 25 2 54 8 66 15 18 10 23 8 32 -13 17 -38 86 -35 152 6 27 17 50 34 50 38 0 16 62 30 85 19 33 -15 72 -2 89 30 8 15 31 43 51 62 35 34 38 35 118 35 77 0 85 2 126 33 24 17 52 32 61 32 9 0 42 18 73 40 30 22 61 40 69 40 21 0 88 -26 100 -38 7 -7 17 -12 24 -12 7 0 35 -11 62 -25 66 -33 263 -84 387 -101 189 -25 372 -12 574 41 106 27 130 37 261 97 41 20 80 37 85 39 6 2 51 31 100 64 166 111 405 372 489 534 10 20 22 43 27 51 5 8 12 22 15 30 3 8 17 40 31 70 54 115 95 313 108 520 13 200 -43 480 -134 672 -28 58 -51 108 -51 112 0 3 -13 24 -29 48 -15 24 -34 60 -40 80 -19 57 3 142 50 193 10 11 22 49 28 85 6 36 16 67 21 68 18 6 31 53 25 83 -4 18 -17 33 -36 41 -16 7 -29 15 -29 18 1 10 38 50 47 50 5 0 20 11 33 25 18 19 22 31 17 61 -3 20 -14 45 -23 55 -16 18 -16 20 6 44 15 16 21 32 18 49 -3 15 1 34 8 43 32 43 7 73 -46 55 l-30 -11 0 85 c0 74 -2 84 -18 84 -21 0 -53 -33 -103 -104 l-34 -48 -5 74 c-7 102 -35 133 -80 88z m-870 -740 c36 -7 75 -14 88 -16 21 -4 23 -9 16 -37 -3 -18 -14 -43 -24 -57 -10 -14 -20 -35 -24 -46 -4 -12 -16 -32 -27 -45 -12 -13 -37 -49 -56 -79 -20 -30 -52 -73 -72 -96 -53 -60 -114 -133 -156 -189 -21 -27 -44 -54 -52 -58 -7 -4 -13 -14 -13 -22 0 -7 -18 -33 -40 -57 -22 -23 -40 -46 -40 -50 0 -5 -19 -21 -42 -38 -47 -35 -85 -38 -188 -15 -115 25 -173 20 -264 -23 -45 -22 -106 -46 -136 -56 -48 -15 -77 -25 -140 -50 -70 -28 -100 -77 -51 -84 14 -2 34 -10 45 -17 12 -7 53 -16 91 -20 90 -9 131 -22 178 -57 20 -16 52 -35 70 -43 18 -7 40 -22 49 -32 16 -18 15 -22 -24 -88 -23 -39 -47 -74 -53 -80 -7 -5 -23 -26 -36 -45 -26 -39 -92 -113 -207 -232 -4 -4 -37 -36 -73 -71 l-66 -64 -20 41 c-58 119 -105 240 -115 301 -40 244 -35 409 20 595 8 30 21 66 28 80 7 14 24 54 38 89 15 35 35 75 46 89 11 13 20 31 20 38 0 8 3 14 8 14 4 0 16 16 27 36 24 45 221 245 278 281 23 15 44 30 47 33 20 20 138 78 250 123 61 24 167 50 250 61 60 7 302 -1 370 -14z m837 -661 c52 -101 102 -279 106 -379 2 -42 0 -45 -28 -51 -16 -4 -101 -7 -187 -8 -166 -1 -229 10 -271 49 -19 19 -19 19 14 49 22 21 44 31 65 31 41 0 84 34 84 66 0 30 12 55 56 112 19 25 37 65 44 95 11 51 53 111 74 104 6 -2 25 -32 43 -68z m-662 -810 c17 -10 40 -24 53 -30 12 -7 22 -16 22 -20 0 -4 17 -13 38 -19 20 -7 44 -18 52 -24 8 -7 33 -21 55 -31 22 -11 42 -23 45 -26 11 -14 109 -49 164 -58 62 -11 101 -7 126 14 15 14 38 18 78 16 39 -2 26 -41 -49 -146 -78 -109 -85 -118 -186 -219 -61 -61 -239 -189 -281 -203 -17 -5 -73 -29 -104 -44 -187 -92 -605 -103 -791 -21 -42 19 -47 24 -37 41 5 11 28 32 51 48 22 15 51 38 64 51 13 12 28 22 33 22 17 0 242 233 242 250 0 6 5 10 10 10 6 0 10 6 10 14 0 25 50 55 100 62 59 8 56 6 115 83 50 66 74 117 75 162 0 14 7 40 16 57 18 38 52 41 99 11z"/> | ||||||
|  | </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 4.8 KiB | 
							
								
								
									
										31
									
								
								my-blog/static/index.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								my-blog/static/index.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | [ | ||||||
|  |   { | ||||||
|  |     "categories": [], | ||||||
|  |     "contents": "ブログを移行しました。過去のブログはsyui.github.ioにありあます。 gh-pagesからcf-pagesへの移行になります。 自作のailogでbuildしています。 特徴としては、atproto, AIとの連携です。 name: Deploy to Cloudflare Pages on: push: branches: - main workflow_dispatch: jobs: deploy: runs-on: ubuntu-latest permissions: contents: read deployments: write steps: - name: Checkout uses: actions/checkout@v4 - name: Setup Rust uses: actions-rs/toolchain@v1 with: toolchain: stable - name: Build ailog run: cargo build --release - name: Build site with ailog run: | cd my-blog ../target/release/ailog build - name: List public directory run: | ls -la my-blog/public/ - name: Deploy to Cloudflare Pages uses: cloudflare/pages-action@v1 with: apiToken: ${{ secrets.CLOUDFLARE_API_TOKEN }} accountId: ${{ secrets.CLOUDFLARE_ACCOUNT_ID }} projectName: ${{ secrets.CLOUDFLARE_PROJECT_NAME }} directory: my-blog/public gitHubToken: ${{ secrets.GITHUB_TOKEN }} wranglerVersion: '3' url https://syui.pages.dev https://syui.github.io", | ||||||
|  |     "description": "ブログを移行しました。過去のブログはsyui.github.ioにありあます。 \n\ngh-pagesからcf-pagesへの移行になります。\n自作のailogでbuildしています。\n特徴としては、atproto, AIとの連携です。\n\nname: Deploy to Cloudflare Pages\n\non:\n  push:\n    branches:\n      - main\n  workfl...", | ||||||
|  |     "formated_time": "Sat Jun 14, 2025", | ||||||
|  |     "href": "https://syui.ai/posts/2025-06-14-blog.html", | ||||||
|  |     "tags": [ | ||||||
|  |       "blog", | ||||||
|  |       "cloudflare", | ||||||
|  |       "github" | ||||||
|  |     ], | ||||||
|  |     "title": "ブログを移行した", | ||||||
|  |     "utc_time": "2025-06-14T00:00:00Z" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "categories": [], | ||||||
|  |     "contents": "rustで静的サイトジェネレータを作りました。ailogといいます。hugoからの移行になります。 ailogは、最初にatproto-comment-system(oauth)とask-AIという機能をつけました。 quick start $ git clone https://git.syui.ai/ai/log $ cd log $ cargo build $ ./target/debug/ailog init my-blog $ ./target/debug/ailog serve my-blog install $ cargo install --path . --- $ export CARGO_HOME="$HOME/.cargo" $ export RUSTUP_HOME="$HOME/.rustup" $ export PATH="$HOME/.cargo/bin:$PATH" --- $ which ailog $ ailog -h build deploy $ cd my-blog $ vim config.toml $ ailog new test $ vim content/posts/`date +"%Y-%m-%d"`.md $ ailog build # publicの中身をweb-serverにdeploy $ cp -rf ./public/* ./web-server/root/ atproto-comment-system example $ cd ./oauth $ npm i $ npm run build $ npm run preview # Production environment variables VITE_APP_HOST=https://example.com VITE_OAUTH_CLIENT_ID=https://example.com/client-metadata.json VITE_OAUTH_REDIRECT_URI=https://example.com/oauth/callback VITE_ADMIN_DID=did:plc:uqzpqmrjnptsxezjx4xuh2mn # Collection names for OAuth app VITE_COLLECTION_COMMENT=ai.syui.log VITE_COLLECTION_USER=ai.syui.log.user VITE_COLLECTION_CHAT=ai.syui.log.chat # Collection names for ailog (backward compatibility) AILOG_COLLECTION_COMMENT=ai.syui.log AILOG_COLLECTION_USER=ai.syui.log.user # API Configuration VITE_BSKY_PUBLIC_API=https://public.api.bsky.app これはailog oauth build my-blogで./my-blog/config.tomlから./oauth/.env.productionが生成されます。 $ ailog oauth build my-blog use 簡単に説明すると、./oauthで生成するのがatproto-comment-systemです。 <script type="module" crossorigin src="/assets/comment-atproto-${hash}}.js"></script> <link rel="stylesheet" crossorigin href="/assets/comment-atproto-${hash}.css"> <section class="comment-section"> <div id="comment-atproto"></div> </section> ただし、oauthであるため、色々と大変です。本番環境(もしくは近い形)でテストを行いましょう。cf, tailscale, ngrokなど。 tunnel: ${hash} credentials-file: ${path}.json ingress: - hostname: example.com service: http://localhost:4173 originRequest: noHappyEyeballs: true - service: http_status:404 # tunnel list, dnsに登録が必要です $ cloudflared tunnel list $ cloudflared tunnel --config cloudflared-config.yml run $ cloudflared tunnel route dns ${uuid} example.com 以下の2つのcollection recordを生成します。ユーザーにはai.syui.logが生成され、ここにコメントが記録されます。それを取得して表示しています。ai.syui.log.userは管理者であるVITE_ADMIN_DID用です。 VITE_COLLECTION_COMMENT=ai.syui.log VITE_COLLECTION_USER=ai.syui.log.user $ ailog auth login $ ailog stream server このコマンドでai.syui.logをjetstreamから監視して、書き込みがあれば、管理者のai.syui.log.userに記録され、そのuser-listに基づいて、コメント一覧を取得します。 つまり、コメント表示のアカウントを手動で設定するか、自動化するか。自動化するならserverでailog stream serverを動かさなければいけません。 ask-AI ask-AIの仕組みは割愛します。後に変更される可能性が高いと思います。 local llm, mcp, atprotoと組み合わせです。 code syntax # comment d=${0:a:h} // This is a comment fn main() { println!("Hello, world!"); } // This is a comment console.log("Hello, world!");", | ||||||
|  |     "description": "rustで静的サイトジェネレータを作りました。ailogといいます。hugoからの移行になります。 \nailogは、最初にatproto-comment-system(oauth)とask-AIという機能をつけました。 \nquick start\n$ git clone https://git.syui.ai/ai/log\n$ cd log\n$ cargo build\n$ ./target/debu...", | ||||||
|  |     "formated_time": "Thu Jun 12, 2025", | ||||||
|  |     "href": "https://syui.ai/posts/2025-06-06-ailog.html", | ||||||
|  |     "tags": [ | ||||||
|  |       "blog", | ||||||
|  |       "rust", | ||||||
|  |       "mcp", | ||||||
|  |       "atp" | ||||||
|  |     ], | ||||||
|  |     "title": "静的サイトジェネレータを作った", | ||||||
|  |     "utc_time": "2025-06-12T00:00:00Z" | ||||||
|  |   } | ||||||
|  | ] | ||||||
							
								
								
									
										295
									
								
								my-blog/static/js/ask-ai.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										295
									
								
								my-blog/static/js/ask-ai.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,295 @@ | |||||||
|  | /** | ||||||
|  |  * Ask AI functionality - Based on original working implementation | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | // Global variables for AI functionality | ||||||
|  | let aiProfileData = null; | ||||||
|  |  | ||||||
|  | // Original functions from working implementation | ||||||
|  | function toggleAskAI() { | ||||||
|  |     const panel = document.getElementById('askAiPanel'); | ||||||
|  |     const isVisible = panel.style.display !== 'none'; | ||||||
|  |     panel.style.display = isVisible ? 'none' : 'block'; | ||||||
|  |      | ||||||
|  |     if (!isVisible) { | ||||||
|  |         checkAuthenticationStatus(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function checkAuthenticationStatus() { | ||||||
|  |     const userSections = document.querySelectorAll('.user-section'); | ||||||
|  |     const isAuthenticated = userSections.length > 0; | ||||||
|  |      | ||||||
|  |     if (isAuthenticated) { | ||||||
|  |         // User is authenticated - show Ask AI UI | ||||||
|  |         document.getElementById('authCheck').style.display = 'none'; | ||||||
|  |         document.getElementById('chatForm').style.display = 'block'; | ||||||
|  |         document.getElementById('chatHistory').style.display = 'block'; | ||||||
|  |          | ||||||
|  |         // Show initial greeting if chat history is empty | ||||||
|  |         const chatHistory = document.getElementById('chatHistory'); | ||||||
|  |         if (chatHistory.children.length === 0) { | ||||||
|  |             showInitialGreeting(); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // Focus on input | ||||||
|  |         setTimeout(() => { | ||||||
|  |             document.getElementById('aiQuestion').focus(); | ||||||
|  |         }, 50); | ||||||
|  |     } else { | ||||||
|  |         // User not authenticated - show auth message | ||||||
|  |         document.getElementById('authCheck').style.display = 'block'; | ||||||
|  |         document.getElementById('chatForm').style.display = 'none'; | ||||||
|  |         document.getElementById('chatHistory').style.display = 'none'; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function askQuestion() { | ||||||
|  |     const question = document.getElementById('aiQuestion').value; | ||||||
|  |     if (!question.trim()) return; | ||||||
|  |      | ||||||
|  |     const askButton = document.getElementById('askButton'); | ||||||
|  |     askButton.disabled = true; | ||||||
|  |     askButton.textContent = 'Posting...'; | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |         // Add user message to chat | ||||||
|  |         addUserMessage(question); | ||||||
|  |          | ||||||
|  |         // Clear input | ||||||
|  |         document.getElementById('aiQuestion').value = ''; | ||||||
|  |          | ||||||
|  |         // Show loading | ||||||
|  |         showLoadingMessage(); | ||||||
|  |          | ||||||
|  |         // Post question via OAuth app | ||||||
|  |         window.dispatchEvent(new CustomEvent('postAIQuestion', { | ||||||
|  |             detail: { question: question } | ||||||
|  |         })); | ||||||
|  |          | ||||||
|  |     } catch (error) { | ||||||
|  |         console.error('Failed to ask question:', error); | ||||||
|  |         showErrorMessage('Sorry, I encountered an error. Please try again.'); | ||||||
|  |     } finally { | ||||||
|  |         askButton.disabled = false; | ||||||
|  |         askButton.textContent = 'Ask'; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function addUserMessage(question) { | ||||||
|  |     const chatHistory = document.getElementById('chatHistory'); | ||||||
|  |     const userSection = document.querySelector('.user-section'); | ||||||
|  |      | ||||||
|  |     let userAvatar = '👤'; | ||||||
|  |     let userDisplay = 'You'; | ||||||
|  |     let userHandle = 'user'; | ||||||
|  |      | ||||||
|  |     if (userSection) { | ||||||
|  |         const avatarImg = userSection.querySelector('.user-avatar'); | ||||||
|  |         const displayName = userSection.querySelector('.user-display-name'); | ||||||
|  |         const handle = userSection.querySelector('.user-handle'); | ||||||
|  |          | ||||||
|  |         if (avatarImg && avatarImg.src) { | ||||||
|  |             userAvatar = `<img src="${avatarImg.src}" alt="${displayName?.textContent || 'User'}" class="profile-avatar">`; | ||||||
|  |         } | ||||||
|  |         if (displayName?.textContent) { | ||||||
|  |             userDisplay = displayName.textContent; | ||||||
|  |         } | ||||||
|  |         if (handle?.textContent) { | ||||||
|  |             userHandle = handle.textContent.replace('@', ''); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const questionDiv = document.createElement('div'); | ||||||
|  |     questionDiv.className = 'chat-message user-message comment-style'; | ||||||
|  |     questionDiv.innerHTML = ` | ||||||
|  |         <div class="message-header"> | ||||||
|  |             <div class="avatar">${userAvatar}</div> | ||||||
|  |             <div class="user-info"> | ||||||
|  |                 <div class="display-name">${userDisplay}</div> | ||||||
|  |                 <div class="handle">@${userHandle}</div> | ||||||
|  |                 <div class="timestamp">${new Date().toLocaleString()}</div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="message-content">${question}</div> | ||||||
|  |     `; | ||||||
|  |     chatHistory.appendChild(questionDiv); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function showLoadingMessage() { | ||||||
|  |     const chatHistory = document.getElementById('chatHistory'); | ||||||
|  |     const loadingDiv = document.createElement('div'); | ||||||
|  |     loadingDiv.className = 'ai-loading-simple'; | ||||||
|  |     loadingDiv.innerHTML = ` | ||||||
|  |         <i class="fas fa-robot"></i> | ||||||
|  |         <span>考えています</span> | ||||||
|  |         <i class="fas fa-spinner fa-spin"></i> | ||||||
|  |     `; | ||||||
|  |     chatHistory.appendChild(loadingDiv); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function showErrorMessage(message) { | ||||||
|  |     const chatHistory = document.getElementById('chatHistory'); | ||||||
|  |     removeLoadingMessage(); | ||||||
|  |      | ||||||
|  |     const errorDiv = document.createElement('div'); | ||||||
|  |     errorDiv.className = 'chat-message error-message comment-style'; | ||||||
|  |     errorDiv.innerHTML = ` | ||||||
|  |         <div class="message-header"> | ||||||
|  |             <div class="avatar">⚠️</div> | ||||||
|  |             <div class="user-info"> | ||||||
|  |                 <div class="display-name">System</div> | ||||||
|  |                 <div class="handle">@system</div> | ||||||
|  |                 <div class="timestamp">${new Date().toLocaleString()}</div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="message-content">${message}</div> | ||||||
|  |     `; | ||||||
|  |     chatHistory.appendChild(errorDiv); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function removeLoadingMessage() { | ||||||
|  |     const loadingMsg = document.querySelector('.ai-loading-simple'); | ||||||
|  |     if (loadingMsg) { | ||||||
|  |         loadingMsg.remove(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function showInitialGreeting() { | ||||||
|  |     if (!aiProfileData) return; | ||||||
|  |  | ||||||
|  |     const chatHistory = document.getElementById('chatHistory'); | ||||||
|  |     const greetingDiv = document.createElement('div'); | ||||||
|  |     greetingDiv.className = 'chat-message ai-message comment-style initial-greeting'; | ||||||
|  |      | ||||||
|  |     const avatarElement = aiProfileData.avatar  | ||||||
|  |         ? `<img src="${aiProfileData.avatar}" alt="${aiProfileData.displayName}" class="profile-avatar">` | ||||||
|  |         : '🤖'; | ||||||
|  |      | ||||||
|  |     greetingDiv.innerHTML = ` | ||||||
|  |         <div class="message-header"> | ||||||
|  |             <div class="avatar">${avatarElement}</div> | ||||||
|  |             <div class="user-info"> | ||||||
|  |                 <div class="display-name">${aiProfileData.displayName}</div> | ||||||
|  |                 <div class="handle">@${aiProfileData.handle}</div> | ||||||
|  |                 <div class="timestamp">${new Date().toLocaleString()}</div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="message-content"> | ||||||
|  |             Hello! I'm an AI assistant trained on this blog's content. I can answer questions about the articles, provide insights, and help you understand the topics discussed here. What would you like to know? | ||||||
|  |         </div> | ||||||
|  |     `; | ||||||
|  |     chatHistory.appendChild(greetingDiv); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function updateAskAIButton() { | ||||||
|  |     const button = document.getElementById('askAiButton'); | ||||||
|  |     if (!button) return; | ||||||
|  |      | ||||||
|  |     // Only update text, never modify the icon | ||||||
|  |     if (aiProfileData && aiProfileData.displayName) { | ||||||
|  |         const textNode = button.childNodes[2] || button.lastChild; | ||||||
|  |         if (textNode && textNode.nodeType === Node.TEXT_NODE) { | ||||||
|  |             textNode.textContent = aiProfileData.displayName; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function handleAIResponse(responseData) { | ||||||
|  |     const chatHistory = document.getElementById('chatHistory'); | ||||||
|  |     removeLoadingMessage(); | ||||||
|  |      | ||||||
|  |     const aiProfile = responseData.aiProfile; | ||||||
|  |     if (!aiProfile || !aiProfile.handle || !aiProfile.displayName) { | ||||||
|  |         console.error('AI profile data is missing'); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const timestamp = new Date(responseData.timestamp || Date.now()); | ||||||
|  |     const avatarElement = aiProfile.avatar  | ||||||
|  |         ? `<img src="${aiProfile.avatar}" alt="${aiProfile.displayName}" class="profile-avatar">` | ||||||
|  |         : '🤖'; | ||||||
|  |      | ||||||
|  |     const answerDiv = document.createElement('div'); | ||||||
|  |     answerDiv.className = 'chat-message ai-message comment-style'; | ||||||
|  |     answerDiv.innerHTML = ` | ||||||
|  |         <div class="message-header"> | ||||||
|  |             <div class="avatar">${avatarElement}</div> | ||||||
|  |             <div class="user-info"> | ||||||
|  |                 <div class="display-name">${aiProfile.displayName}</div> | ||||||
|  |                 <div class="handle">@${aiProfile.handle}</div> | ||||||
|  |                 <div class="timestamp">${timestamp.toLocaleString()}</div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |         <div class="message-content">${responseData.answer}</div> | ||||||
|  |     `; | ||||||
|  |     chatHistory.appendChild(answerDiv); | ||||||
|  |      | ||||||
|  |     // Limit chat history | ||||||
|  |     limitChatHistory(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function limitChatHistory() { | ||||||
|  |     const chatHistory = document.getElementById('chatHistory'); | ||||||
|  |     if (chatHistory.children.length > 10) { | ||||||
|  |         chatHistory.removeChild(chatHistory.children[0]); | ||||||
|  |         if (chatHistory.children.length > 0) { | ||||||
|  |             chatHistory.removeChild(chatHistory.children[0]); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Event listeners setup | ||||||
|  | function setupAskAIEventListeners() { | ||||||
|  |     // Listen for AI profile updates from OAuth app | ||||||
|  |     window.addEventListener('aiProfileLoaded', function(event) { | ||||||
|  |         aiProfileData = event.detail; | ||||||
|  |         console.log('AI profile loaded:', aiProfileData); | ||||||
|  |         updateAskAIButton(); | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     // Listen for AI responses | ||||||
|  |     window.addEventListener('aiResponseReceived', function(event) { | ||||||
|  |         handleAIResponse(event.detail); | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     // Track IME composition state | ||||||
|  |     let isComposing = false; | ||||||
|  |     const aiQuestionInput = document.getElementById('aiQuestion'); | ||||||
|  |      | ||||||
|  |     if (aiQuestionInput) { | ||||||
|  |         aiQuestionInput.addEventListener('compositionstart', function() { | ||||||
|  |             isComposing = true; | ||||||
|  |         }); | ||||||
|  |          | ||||||
|  |         aiQuestionInput.addEventListener('compositionend', function() { | ||||||
|  |             isComposing = false; | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Keyboard shortcuts | ||||||
|  |     document.addEventListener('keydown', function(e) { | ||||||
|  |         if (e.key === 'Escape') { | ||||||
|  |             const panel = document.getElementById('askAiPanel'); | ||||||
|  |             if (panel) { | ||||||
|  |                 panel.style.display = 'none'; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // Enter key to send message (only when not composing Japanese input) | ||||||
|  |         if (e.key === 'Enter' && e.target.id === 'aiQuestion' && !e.shiftKey && !isComposing) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             askQuestion(); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Initialize Ask AI when DOM is loaded | ||||||
|  | document.addEventListener('DOMContentLoaded', function() { | ||||||
|  |     setupAskAIEventListeners(); | ||||||
|  |     console.log('Ask AI initialized successfully'); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Global functions for onclick handlers | ||||||
|  | window.toggleAskAI = toggleAskAI; | ||||||
|  | window.askQuestion = askQuestion; | ||||||
							
								
								
									
										94
									
								
								my-blog/static/js/theme.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								my-blog/static/js/theme.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,94 @@ | |||||||
|  | /** | ||||||
|  |  * Theme and visual effects - Pure CSS animations, no jQuery | ||||||
|  |  */ | ||||||
|  | class Theme { | ||||||
|  |     constructor() { | ||||||
|  |         this.init(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     init() { | ||||||
|  |         this.setupParticleColors(); | ||||||
|  |         this.setupLogoAnimations(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     setupParticleColors() { | ||||||
|  |         // Dynamic particle colors based on theme | ||||||
|  |         const style = document.createElement('style'); | ||||||
|  |         style.textContent = ` | ||||||
|  |             /* Dynamic particle colors based on theme */ | ||||||
|  |             .likeButton .particleLayer circle:nth-child(1), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(2) { | ||||||
|  |                 fill: var(--particle-color-1) !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton .particleLayer circle:nth-child(3), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(4) { | ||||||
|  |                 fill: var(--particle-color-2) !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton .particleLayer circle:nth-child(5), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(6), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(7) { | ||||||
|  |                 fill: var(--particle-color-3) !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton .particleLayer circle:nth-child(8), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(9), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(10) { | ||||||
|  |                 fill: var(--particle-color-4) !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton .particleLayer circle:nth-child(11), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(12), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(13), | ||||||
|  |             .likeButton .particleLayer circle:nth-child(14) { | ||||||
|  |                 fill: var(--particle-color-5) !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             /* Reset initial animations but allow hover */ | ||||||
|  |             .likeButton .syui { | ||||||
|  |                 animation: none; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton .particleLayer { | ||||||
|  |                 animation: none;  | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton .explosion { | ||||||
|  |                 animation: none; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             /* Enable hover animations from package */ | ||||||
|  |             .likeButton:hover .syui, | ||||||
|  |             .likeButton:hover path.syui { | ||||||
|  |                 animation: syuiDeluxeAnime 400ms forwards !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton:hover .particleLayer { | ||||||
|  |                 animation: particleLayerAnime 800ms forwards !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             .likeButton:hover .explosion { | ||||||
|  |                 animation: explosionAnime 800ms forwards !important; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             /* Logo positioning */ | ||||||
|  |             .logo .likeButton { | ||||||
|  |                 background: transparent !important; | ||||||
|  |                 display: block; | ||||||
|  |             } | ||||||
|  |         `; | ||||||
|  |         document.head.appendChild(style); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     setupLogoAnimations() { | ||||||
|  |         // Pure CSS animations are handled by the svg-animation-package.css | ||||||
|  |         // This method is reserved for any future JavaScript-based enhancements | ||||||
|  |         console.log('Logo animations initialized (CSS-based)'); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Initialize theme when DOM is loaded | ||||||
|  | document.addEventListener('DOMContentLoaded', () => { | ||||||
|  |     new Theme(); | ||||||
|  | }); | ||||||
							
								
								
									
										165
									
								
								my-blog/static/pkg/font-awesome/LICENSE.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										165
									
								
								my-blog/static/pkg/font-awesome/LICENSE.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,165 @@ | |||||||
|  | Fonticons, Inc. (https://fontawesome.com) | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | Font Awesome Free License | ||||||
|  |  | ||||||
|  | Font Awesome Free is free, open source, and GPL friendly. You can use it for | ||||||
|  | commercial projects, open source projects, or really almost whatever you want. | ||||||
|  | Full Font Awesome Free license: https://fontawesome.com/license/free. | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | # Icons: CC BY 4.0 License (https://creativecommons.org/licenses/by/4.0/) | ||||||
|  |  | ||||||
|  | The Font Awesome Free download is licensed under a Creative Commons | ||||||
|  | Attribution 4.0 International License and applies to all icons packaged | ||||||
|  | as SVG and JS file types. | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | # Fonts: SIL OFL 1.1 License | ||||||
|  |  | ||||||
|  | In the Font Awesome Free download, the SIL OFL license applies to all icons | ||||||
|  | packaged as web and desktop font files. | ||||||
|  |  | ||||||
|  | Copyright (c) 2024 Fonticons, Inc. (https://fontawesome.com) | ||||||
|  | with Reserved Font Name: "Font Awesome". | ||||||
|  |  | ||||||
|  | This Font Software is licensed under the SIL Open Font License, Version 1.1. | ||||||
|  | This license is copied below, and is also available with a FAQ at: | ||||||
|  | http://scripts.sil.org/OFL | ||||||
|  |  | ||||||
|  | SIL OPEN FONT LICENSE | ||||||
|  | Version 1.1 - 26 February 2007 | ||||||
|  |  | ||||||
|  | PREAMBLE | ||||||
|  | The goals of the Open Font License (OFL) are to stimulate worldwide | ||||||
|  | development of collaborative font projects, to support the font creation | ||||||
|  | efforts of academic and linguistic communities, and to provide a free and | ||||||
|  | open framework in which fonts may be shared and improved in partnership | ||||||
|  | with others. | ||||||
|  |  | ||||||
|  | The OFL allows the licensed fonts to be used, studied, modified and | ||||||
|  | redistributed freely as long as they are not sold by themselves. The | ||||||
|  | fonts, including any derivative works, can be bundled, embedded, | ||||||
|  | redistributed and/or sold with any software provided that any reserved | ||||||
|  | names are not used by derivative works. The fonts and derivatives, | ||||||
|  | however, cannot be released under any other type of license. The | ||||||
|  | requirement for fonts to remain under this license does not apply | ||||||
|  | to any document created using the fonts or their derivatives. | ||||||
|  |  | ||||||
|  | DEFINITIONS | ||||||
|  | "Font Software" refers to the set of files released by the Copyright | ||||||
|  | Holder(s) under this license and clearly marked as such. This may | ||||||
|  | include source files, build scripts and documentation. | ||||||
|  |  | ||||||
|  | "Reserved Font Name" refers to any names specified as such after the | ||||||
|  | copyright statement(s). | ||||||
|  |  | ||||||
|  | "Original Version" refers to the collection of Font Software components as | ||||||
|  | distributed by the Copyright Holder(s). | ||||||
|  |  | ||||||
|  | "Modified Version" refers to any derivative made by adding to, deleting, | ||||||
|  | or substituting — in part or in whole — any of the components of the | ||||||
|  | Original Version, by changing formats or by porting the Font Software to a | ||||||
|  | new environment. | ||||||
|  |  | ||||||
|  | "Author" refers to any designer, engineer, programmer, technical | ||||||
|  | writer or other person who contributed to the Font Software. | ||||||
|  |  | ||||||
|  | PERMISSION & CONDITIONS | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining | ||||||
|  | a copy of the Font Software, to use, study, copy, merge, embed, modify, | ||||||
|  | redistribute, and sell modified and unmodified copies of the Font | ||||||
|  | Software, subject to the following conditions: | ||||||
|  |  | ||||||
|  | 1) Neither the Font Software nor any of its individual components, | ||||||
|  | in Original or Modified Versions, may be sold by itself. | ||||||
|  |  | ||||||
|  | 2) Original or Modified Versions of the Font Software may be bundled, | ||||||
|  | redistributed and/or sold with any software, provided that each copy | ||||||
|  | contains the above copyright notice and this license. These can be | ||||||
|  | included either as stand-alone text files, human-readable headers or | ||||||
|  | in the appropriate machine-readable metadata fields within text or | ||||||
|  | binary files as long as those fields can be easily viewed by the user. | ||||||
|  |  | ||||||
|  | 3) No Modified Version of the Font Software may use the Reserved Font | ||||||
|  | Name(s) unless explicit written permission is granted by the corresponding | ||||||
|  | Copyright Holder. This restriction only applies to the primary font name as | ||||||
|  | presented to the users. | ||||||
|  |  | ||||||
|  | 4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font | ||||||
|  | Software shall not be used to promote, endorse or advertise any | ||||||
|  | Modified Version, except to acknowledge the contribution(s) of the | ||||||
|  | Copyright Holder(s) and the Author(s) or with their explicit written | ||||||
|  | permission. | ||||||
|  |  | ||||||
|  | 5) The Font Software, modified or unmodified, in part or in whole, | ||||||
|  | must be distributed entirely under this license, and must not be | ||||||
|  | distributed under any other license. The requirement for fonts to | ||||||
|  | remain under this license does not apply to any document created | ||||||
|  | using the Font Software. | ||||||
|  |  | ||||||
|  | TERMINATION | ||||||
|  | This license becomes null and void if any of the above conditions are | ||||||
|  | not met. | ||||||
|  |  | ||||||
|  | DISCLAIMER | ||||||
|  | THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||||||
|  | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF | ||||||
|  | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT | ||||||
|  | OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE | ||||||
|  | COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | ||||||
|  | INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL | ||||||
|  | DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | ||||||
|  | FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM | ||||||
|  | OTHER DEALINGS IN THE FONT SOFTWARE. | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | # Code: MIT License (https://opensource.org/licenses/MIT) | ||||||
|  |  | ||||||
|  | In the Font Awesome Free download, the MIT license applies to all non-font and | ||||||
|  | non-icon files. | ||||||
|  |  | ||||||
|  | Copyright 2024 Fonticons, Inc. | ||||||
|  |  | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining a copy of | ||||||
|  | this software and associated documentation files (the "Software"), to deal in the | ||||||
|  | Software without restriction, including without limitation the rights to use, copy, | ||||||
|  | modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, | ||||||
|  | and to permit persons to whom the Software is furnished to do so, subject to the | ||||||
|  | following conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be included in all | ||||||
|  | copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, | ||||||
|  | INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A | ||||||
|  | PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT | ||||||
|  | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION | ||||||
|  | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE | ||||||
|  | SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | # Attribution | ||||||
|  |  | ||||||
|  | Attribution is required by MIT, SIL OFL, and CC BY licenses. Downloaded Font | ||||||
|  | Awesome Free files already contain embedded comments with sufficient | ||||||
|  | attribution, so you shouldn't need to do anything additional when using these | ||||||
|  | files normally. | ||||||
|  |  | ||||||
|  | We've kept attribution comments terse, so we ask that you do not actively work | ||||||
|  | to remove them from files, especially code. They're a great way for folks to | ||||||
|  | learn about Font Awesome. | ||||||
|  |  | ||||||
|  | -------------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | # Brand Icons | ||||||
|  |  | ||||||
|  | All brand icons are trademarks of their respective owners. The use of these | ||||||
|  | trademarks does not indicate endorsement of the trademark holder by Font | ||||||
|  | Awesome, nor vice versa. **Please do not use brand logos for any purpose except | ||||||
|  | to represent the company, product, or service to which they refer.** | ||||||
							
								
								
									
										9
									
								
								my-blog/static/pkg/font-awesome/css/all.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								my-blog/static/pkg/font-awesome/css/all.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										6
									
								
								my-blog/static/pkg/font-awesome/css/brands.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								my-blog/static/pkg/font-awesome/css/brands.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										9
									
								
								my-blog/static/pkg/font-awesome/css/fontawesome.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								my-blog/static/pkg/font-awesome/css/fontawesome.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										6
									
								
								my-blog/static/pkg/font-awesome/css/regular.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								my-blog/static/pkg/font-awesome/css/regular.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | /*! | ||||||
|  |  * Font Awesome Free 6.5.2 by @fontawesome - https://fontawesome.com | ||||||
|  |  * License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) | ||||||
|  |  * Copyright 2024 Fonticons, Inc. | ||||||
|  |  */ | ||||||
|  | :host,:root{--fa-style-family-classic:"Font Awesome 6 Free";--fa-font-regular:normal 400 1em/1 "Font Awesome 6 Free"}@font-face{font-family:"Font Awesome 6 Free";font-style:normal;font-weight:400;font-display:block;src:url(../webfonts/fa-regular-400.woff2) format("woff2"),url(../webfonts/fa-regular-400.ttf) format("truetype")}.fa-regular,.far{font-weight:400} | ||||||
							
								
								
									
										6
									
								
								my-blog/static/pkg/font-awesome/css/solid.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								my-blog/static/pkg/font-awesome/css/solid.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | /*! | ||||||
|  |  * Font Awesome Free 6.5.2 by @fontawesome - https://fontawesome.com | ||||||
|  |  * License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) | ||||||
|  |  * Copyright 2024 Fonticons, Inc. | ||||||
|  |  */ | ||||||
|  | :host,:root{--fa-style-family-classic:"Font Awesome 6 Free";--fa-font-solid:normal 900 1em/1 "Font Awesome 6 Free"}@font-face{font-family:"Font Awesome 6 Free";font-style:normal;font-weight:900;font-display:block;src:url(../webfonts/fa-solid-900.woff2) format("woff2"),url(../webfonts/fa-solid-900.ttf) format("truetype")}.fa-solid,.fas{font-weight:900} | ||||||
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-brands-400.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-brands-400.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-brands-400.woff2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-brands-400.woff2
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-regular-400.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-regular-400.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-regular-400.woff2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-regular-400.woff2
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-solid-900.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-solid-900.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-solid-900.woff2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-solid-900.woff2
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-v4compatibility.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/font-awesome/webfonts/fa-v4compatibility.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.eot
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.eot
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										34
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.svg
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| After Width: | Height: | Size: 58 KiB | 
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.woff
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								my-blog/static/pkg/icomoon/fonts/icomoon.woff
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										99
									
								
								my-blog/static/pkg/icomoon/style.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								my-blog/static/pkg/icomoon/style.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | |||||||
|  | @font-face { | ||||||
|  |   font-family: 'icomoon'; | ||||||
|  |   src:  url('fonts/icomoon.eot?mxezzh'); | ||||||
|  |   src:  url('fonts/icomoon.eot?mxezzh#iefix') format('embedded-opentype'), | ||||||
|  |     url('fonts/icomoon.ttf?mxezzh') format('truetype'), | ||||||
|  |     url('fonts/icomoon.woff?mxezzh') format('woff'), | ||||||
|  |     url('fonts/icomoon.svg?mxezzh#icomoon') format('svg'); | ||||||
|  |   font-weight: normal; | ||||||
|  |   font-style: normal; | ||||||
|  |   font-display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [class^="icon-"], [class*=" icon-"] { | ||||||
|  |   /* use !important to prevent issues with browser extensions that change fonts */ | ||||||
|  |   font-family: 'icomoon' !important; | ||||||
|  |   speak: never; | ||||||
|  |   font-style: normal; | ||||||
|  |   font-weight: normal; | ||||||
|  |   font-variant: normal; | ||||||
|  |   text-transform: none; | ||||||
|  |   line-height: 1; | ||||||
|  |  | ||||||
|  |   /* Better Font Rendering =========== */ | ||||||
|  |   -webkit-font-smoothing: antialiased; | ||||||
|  |   -moz-osx-font-smoothing: grayscale; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .icon-git:before { | ||||||
|  |   content: "\e901"; | ||||||
|  | } | ||||||
|  | .icon-cube:before { | ||||||
|  |   content: "\e900"; | ||||||
|  | } | ||||||
|  | .icon-game:before { | ||||||
|  |   content: "\e9d5"; | ||||||
|  | } | ||||||
|  | .icon-card:before { | ||||||
|  |   content: "\e9d6"; | ||||||
|  | } | ||||||
|  | .icon-book:before { | ||||||
|  |   content: "\e9d7"; | ||||||
|  | } | ||||||
|  | .icon-git1:before { | ||||||
|  |   content: "\e9d3"; | ||||||
|  | } | ||||||
|  | .icon-moji_a:before { | ||||||
|  |   content: "\e9c3"; | ||||||
|  | } | ||||||
|  | .icon-archlinux:before { | ||||||
|  |   content: "\e9c4"; | ||||||
|  | } | ||||||
|  | .icon-archlinuxjp:before { | ||||||
|  |   content: "\e9c5"; | ||||||
|  | } | ||||||
|  | .icon-syui:before { | ||||||
|  |   content: "\e9c6"; | ||||||
|  | } | ||||||
|  | .icon-phoenix-power:before { | ||||||
|  |   content: "\e9c7"; | ||||||
|  | } | ||||||
|  | .icon-phoenix-world:before { | ||||||
|  |   content: "\e9c8"; | ||||||
|  | } | ||||||
|  | .icon-power:before { | ||||||
|  |   content: "\e9c9"; | ||||||
|  | } | ||||||
|  | .icon-phoenix:before { | ||||||
|  |   content: "\e9ca"; | ||||||
|  | } | ||||||
|  | .icon-honeycomb:before { | ||||||
|  |   content: "\e9cb"; | ||||||
|  | } | ||||||
|  | .icon-ai:before { | ||||||
|  |   content: "\e9cc"; | ||||||
|  | } | ||||||
|  | .icon-robot:before { | ||||||
|  |   content: "\e9cd"; | ||||||
|  | } | ||||||
|  | .icon-sandar:before { | ||||||
|  |   content: "\e9ce"; | ||||||
|  | } | ||||||
|  | .icon-moon:before { | ||||||
|  |   content: "\e9cf"; | ||||||
|  | } | ||||||
|  | .icon-home:before { | ||||||
|  |   content: "\e9d0"; | ||||||
|  | } | ||||||
|  | .icon-cloud:before { | ||||||
|  |   content: "\e9d1"; | ||||||
|  | } | ||||||
|  | .icon-api:before { | ||||||
|  |   content: "\e9d2"; | ||||||
|  | } | ||||||
|  | .icon-aibadge:before { | ||||||
|  |   content: "\ebf8"; | ||||||
|  | } | ||||||
|  | .icon-aiterm:before { | ||||||
|  |   content: "\ebf7"; | ||||||
|  | } | ||||||
							
								
								
									
										24
									
								
								my-blog/static/syui.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								my-blog/static/syui.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | <svg width="77pt" height="77pt" viewBox="0 0 512 512" class="likeButton" > | ||||||
|  |   <circle class="explosion" r="150" cx="250" cy="250"></circle> | ||||||
|  |   <g class="particleLayer"> | ||||||
|  |     <circle fill="#ef454aba" cx="130" cy="126.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454acc" cx="411" cy="313.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="279" cy="86.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="155" cy="390.5" r="12.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="89" cy="292.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="414" cy="282.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454a91" cx="115" cy="149.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="250" cy="80.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="78" cy="261.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454a91" cx="182" cy="402.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="401.5" cy="166" r="13"/> | ||||||
|  |     <circle fill="#ef454aba" cx="379" cy="141.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454a91" cx="327" cy="397.5" r="10.5"/> | ||||||
|  |     <circle fill="#ef454aba" cx="296" cy="392.5" r="10.5"/> | ||||||
|  |   </g> | ||||||
|  | <g transform="translate(0,512) scale(0.1,-0.1)" fill="#000000" class="icon_syui"> | ||||||
|  |  | ||||||
|  | <path class="syui" d="M3660 4460 c-11 -11 -33 -47 -48 -80 l-29 -60 -12 38 c-27 88 -58 92 -98 11 -35 -70 -73 -159 -73 -169 0 -6 -5 -10 -10 -10 -6 0 -15 -10 -21 -22 -33 -73 -52 -92 -47 -48 2 26 -1 35 -14 38 -16 3 -168 -121 -168 -138 0 -5 -13 -16 -28 -24 -24 -13 -35 -12 -87 0 -221 55 -231 56 -480 56 -219 1 -247 -1 -320 -22 -44 -12 -96 -26 -115 -30 -57 -13 -122 -39 -200 -82 -8 -4 -31 -14 -50 -23 -41 -17 -34 -13 -146 -90 -87 -59 -292 -252 -351 -330 -63 -83 -143 -209 -143 -225 0 -10 -7 -23 -15 -30 -8 -7 -15 -17 -15 -22 0 -5 -13 -37 -28 -71 -16 -34 -36 -93 -45 -132 -9 -38 -24 -104 -34 -145 -13 -60 -17 -121 -17 -300 1 -224 1 -225 36 -365 24 -94 53 -175 87 -247 28 -58 51 -108 51 -112 0 -3 13 -24 28 -48 42 -63 46 -79 22 -85 -11 -3 -20 -9 -20 -14 0 -5 -4 -9 -10 -9 -5 0 -22 -11 -37 -25 -16 -13 -75 -59 -133 -100 -58 -42 -113 -82 -123 -90 -9 -8 -22 -15 -27 -15 -6 0 -10 -6 -10 -13 0 -8 -11 -20 -25 -27 -34 -18 -34 -54 0 -48 14 3 25 2 25 -1 0 -3 -43 -31 -95 -61 -52 -30 -95 -58 -95 -62 0 -5 -5 -8 -11 -8 -19 0 -84 -33 -92 -47 -4 -7 -15 -13 -22 -13 -14 0 -17 -4 -19 -32 -1 -8 15 -15 37 -18 l38 -5 -47 -48 c-56 -59 -54 -81 9 -75 30 3 45 0 54 -11 9 -13 16 -14 43 -4 29 11 30 10 18 -5 -7 -9 -19 -23 -25 -30 -7 -7 -13 -20 -13 -29 0 -12 8 -14 38 -9 20 4 57 8 82 9 25 2 54 8 66 15 18 10 23 8 32 -13 17 -38 86 -35 152 6 27 17 50 34 50 38 0 16 62 30 85 19 33 -15 72 -2 89 30 8 15 31 43 51 62 35 34 38 35 118 35 77 0 85 2 126 33 24 17 52 32 61 32 9 0 42 18 73 40 30 22 61 40 69 40 21 0 88 -26 100 -38 7 -7 17 -12 24 -12 7 0 35 -11 62 -25 66 -33 263 -84 387 -101 189 -25 372 -12 574 41 106 27 130 37 261 97 41 20 80 37 85 39 6 2 51 31 100 64 166 111 405 372 489 534 10 20 22 43 27 51 5 8 12 22 15 30 3 8 17 40 31 70 54 115 95 313 108 520 13 200 -43 480 -134 672 -28 58 -51 108 -51 112 0 3 -13 24 -29 48 -15 24 -34 60 -40 80 -19 57 3 142 50 193 10 11 22 49 28 85 6 36 16 67 21 68 18 6 31 53 25 83 -4 18 -17 33 -36 41 -16 7 -29 15 -29 18 1 10 38 50 47 50 5 0 20 11 33 25 18 19 22 31 17 61 -3 20 -14 45 -23 55 -16 18 -16 20 6 44 15 16 21 32 18 49 -3 15 1 34 8 43 32 43 7 73 -46 55 l-30 -11 0 85 c0 74 -2 84 -18 84 -21 0 -53 -33 -103 -104 l-34 -48 -5 74 c-7 102 -35 133 -80 88z m-870 -740 c36 -7 75 -14 88 -16 21 -4 23 -9 16 -37 -3 -18 -14 -43 -24 -57 -10 -14 -20 -35 -24 -46 -4 -12 -16 -32 -27 -45 -12 -13 -37 -49 -56 -79 -20 -30 -52 -73 -72 -96 -53 -60 -114 -133 -156 -189 -21 -27 -44 -54 -52 -58 -7 -4 -13 -14 -13 -22 0 -7 -18 -33 -40 -57 -22 -23 -40 -46 -40 -50 0 -5 -19 -21 -42 -38 -47 -35 -85 -38 -188 -15 -115 25 -173 20 -264 -23 -45 -22 -106 -46 -136 -56 -48 -15 -77 -25 -140 -50 -70 -28 -100 -77 -51 -84 14 -2 34 -10 45 -17 12 -7 53 -16 91 -20 90 -9 131 -22 178 -57 20 -16 52 -35 70 -43 18 -7 40 -22 49 -32 16 -18 15 -22 -24 -88 -23 -39 -47 -74 -53 -80 -7 -5 -23 -26 -36 -45 -26 -39 -92 -113 -207 -232 -4 -4 -37 -36 -73 -71 l-66 -64 -20 41 c-58 119 -105 240 -115 301 -40 244 -35 409 20 595 8 30 21 66 28 80 7 14 24 54 38 89 15 35 35 75 46 89 11 13 20 31 20 38 0 8 3 14 8 14 4 0 16 16 27 36 24 45 221 245 278 281 23 15 44 30 47 33 20 20 138 78 250 123 61 24 167 50 250 61 60 7 302 -1 370 -14z m837 -661 c52 -101 102 -279 106 -379 2 -42 0 -45 -28 -51 -16 -4 -101 -7 -187 -8 -166 -1 -229 10 -271 49 -19 19 -19 19 14 49 22 21 44 31 65 31 41 0 84 34 84 66 0 30 12 55 56 112 19 25 37 65 44 95 11 51 53 111 74 104 6 -2 25 -32 43 -68z m-662 -810 c17 -10 40 -24 53 -30 12 -7 22 -16 22 -20 0 -4 17 -13 38 -19 20 -7 44 -18 52 -24 8 -7 33 -21 55 -31 22 -11 42 -23 45 -26 11 -14 109 -49 164 -58 62 -11 101 -7 126 14 15 14 38 18 78 16 39 -2 26 -41 -49 -146 -78 -109 -85 -118 -186 -219 -61 -61 -239 -189 -281 -203 -17 -5 -73 -29 -104 -44 -187 -92 -605 -103 -791 -21 -42 19 -47 24 -37 41 5 11 28 32 51 48 22 15 51 38 64 51 13 12 28 22 33 22 17 0 242 233 242 250 0 6 5 10 10 10 6 0 10 6 10 14 0 25 50 55 100 62 59 8 56 6 115 83 50 66 74 117 75 162 0 14 7 40 16 57 18 38 52 41 99 11z"/> | ||||||
|  | </g> | ||||||
|  |  | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 4.8 KiB | 
							
								
								
									
										97
									
								
								my-blog/templates/base.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								my-blog/templates/base.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | |||||||
|  | <!DOCTYPE html> | ||||||
|  | <html lang="{{ config.language }}"> | ||||||
|  | <head> | ||||||
|  |     <meta charset="UTF-8"> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|  |     <title>{% block title %}{{ config.title }}{% endblock %}</title> | ||||||
|  |      | ||||||
|  |     <!-- Favicon --> | ||||||
|  |     <link rel="icon" href="/favicon.ico" /> | ||||||
|  |     <link rel="apple-touch-icon" href="/apple-touch-icon.png" /> | ||||||
|  |      | ||||||
|  |     <!-- Stylesheets --> | ||||||
|  |     <link rel="stylesheet" href="/css/style.css"> | ||||||
|  |     <link rel="stylesheet" href="/css/svg-animation-package.css"> | ||||||
|  |     <link rel="stylesheet" href="/pkg/icomoon/style.css"> | ||||||
|  |     <link rel="stylesheet" href="/pkg/font-awesome/css/all.min.css"> | ||||||
|  |      | ||||||
|  |     {% block head %}{% endblock %} | ||||||
|  | </head> | ||||||
|  | <body> | ||||||
|  |     <div class="container"> | ||||||
|  |         <header class="main-header"> | ||||||
|  |             <div class="header-content"> | ||||||
|  |                 <h1><a href="/" class="site-title">{{ config.title }}</a></h1> | ||||||
|  |                 <div class="logo"> | ||||||
|  |                     <a href="/"> | ||||||
|  |                         <svg width="77pt" height="77pt" viewBox="0 0 512 512" class="likeButton"> | ||||||
|  |                             <circle class="explosion" r="150" cx="250" cy="250"></circle> | ||||||
|  |                             <g class="particleLayer"> | ||||||
|  |                                 <circle fill="#8CE8C3" cx="130" cy="126.5" r="12.5"></circle> | ||||||
|  |                                 <circle fill="#8CE8C3" cx="411" cy="313.5" r="12.5"></circle> | ||||||
|  |                                 <circle fill="#91D2FA" cx="279" cy="86.5" r="12.5"></circle> | ||||||
|  |                                 <circle fill="#91D2FA" cx="155" cy="390.5" r="12.5"></circle> | ||||||
|  |                                 <circle fill="#CC8EF5" cx="89" cy="292.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#9BDFBA" cx="414" cy="282.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#9BDFBA" cx="115" cy="149.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#9FC7FA" cx="250" cy="80.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#9FC7FA" cx="78" cy="261.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#96D8E9" cx="182" cy="402.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#CC8EF5" cx="401.5" cy="166" r="13"></circle> | ||||||
|  |                                 <circle fill="#DB92D0" cx="379" cy="141.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#DB92D0" cx="327" cy="397.5" r="10.5"></circle> | ||||||
|  |                                 <circle fill="#DD99B8" cx="296" cy="392.5" r="10.5"></circle> | ||||||
|  |                             </g> | ||||||
|  |                             <g transform="translate(0,512) scale(0.1,-0.1)" fill="#000000" class="icon_syui"> | ||||||
|  |                                 <path class="syui" d="M3660 4460 c-11 -11 -33 -47 -48 -80 l-29 -60 -12 38 c-27 88 -58 92 -98 11 -35 -70 -73 -159 -73 -169 0 -6 -5 -10 -10 -10 -6 0 -15 -10 -21 -22 -33 -73 -52 -92 -47 -48 2 26 -1 35 -14 38 -16 3 -168 -121 -168 -138 0 -5 -13 -16 -28 -24 -24 -13 -35 -12 -87 0 -221 55 -231 56 -480 56 -219 1 -247 -1 -320 -22 -44 -12 -96 -26 -115 -30 -57 -13 -122 -39 -200 -82 -8 -4 -31 -14 -50 -23 -41 -17 -34 -13 -146 -90 -87 -59 -292 -252 -351 -330 -63 -83 -143 -209 -143 -225 0 -10 -7 -23 -15 -30 -8 -7 -15 -17 -15 -22 0 -5 -13 -37 -28 -71 -16 -34 -36 -93 -45 -132 -9 -38 -24 -104 -34 -145 -13 -60 -17 -121 -17 -300 1 -224 1 -225 36 -365 24 -94 53 -175 87 -247 28 -58 51 -108 51 -112 0 -3 13 -24 28 -48 42 -63 46 -79 22 -85 -11 -3 -20 -9 -20 -14 0 -5 -4 -9 -10 -9 -5 0 -22 -11 -37 -25 -16 -13 -75 -59 -133 -100 -58 -42 -113 -82 -123 -90 -9 -8 -22 -15 -27 -15 -6 0 -10 -6 -10 -13 0 -8 -11 -20 -25 -27 -34 -18 -34 -54 0 -48 14 3 25 2 25 -1 0 -3 -43 -31 -95 -61 -52 -30 -95 -58 -95 -62 0 -5 -5 -8 -11 -8 -19 0 -84 -33 -92 -47 -4 -7 -15 -13 -22 -13 -14 0 -17 -4 -19 -32 -1 -8 15 -15 37 -18 l38 -5 -47 -48 c-56 -59 -54 -81 9 -75 30 3 45 0 54 -11 9 -13 16 -14 43 -4 29 11 30 10 18 -5 -7 -9 -19 -23 -25 -30 -7 -7 -13 -20 -13 -29 0 -12 8 -14 38 -9 20 4 57 8 82 9 25 2 54 8 66 15 18 10 23 8 32 -13 17 -38 86 -35 152 6 27 17 50 34 50 38 0 16 62 30 85 19 33 -15 72 -2 89 30 8 15 31 43 51 62 35 34 38 35 118 35 77 0 85 2 126 33 24 17 52 32 61 32 9 0 42 18 73 40 30 22 61 40 69 40 21 0 88 -26 100 -38 7 -7 17 -12 24 -12 7 0 35 -11 62 -25 66 -33 263 -84 387 -101 189 -25 372 -12 574 41 106 27 130 37 261 97 41 20 80 37 85 39 6 2 51 31 100 64 166 111 405 372 489 534 10 20 22 43 27 51 5 8 12 22 15 30 3 8 17 40 31 70 54 115 95 313 108 520 13 200 -43 480 -134 672 -28 58 -51 108 -51 112 0 3 -13 24 -29 48 -15 24 -34 60 -40 80 -19 57 3 142 50 193 10 11 22 49 28 85 6 36 16 67 21 68 18 6 31 53 25 83 -4 18 -17 33 -36 41 -16 7 -29 15 -29 18 1 10 38 50 47 50 5 0 20 11 33 25 18 19 22 31 17 61 -3 20 -14 45 -23 55 -16 18 -16 20 6 44 15 16 21 32 18 49 -3 15 1 34 8 43 32 43 7 73 -46 55 l-30 -11 0 85 c0 74 -2 84 -18 84 -21 0 -53 -33 -103 -104 l-34 -48 -5 74 c-7 102 -35 133 -80 88z m-870 -740 c36 -7 75 -14 88 -16 21 -4 23 -9 16 -37 -3 -18 -14 -43 -24 -57 -10 -14 -20 -35 -24 -46 -4 -12 -16 -32 -27 -45 -12 -13 -37 -49 -56 -79 -20 -30 -52 -73 -72 -96 -53 -60 -114 -133 -156 -189 -21 -27 -44 -54 -52 -58 -7 -4 -13 -14 -13 -22 0 -7 -18 -33 -40 -57 -22 -23 -40 -46 -40 -50 0 -5 -19 -21 -42 -38 -47 -35 -85 -38 -188 -15 -115 25 -173 20 -264 -23 -45 -22 -106 -46 -136 -56 -48 -15 -77 -25 -140 -50 -70 -28 -100 -77 -51 -84 14 -2 34 -10 45 -17 12 -7 53 -16 91 -20 90 -9 131 -22 178 -57 20 -16 52 -35 70 -43 18 -7 40 -22 49 -32 16 -18 15 -22 -24 -88 -23 -39 -47 -74 -53 -80 -7 -5 -23 -26 -36 -45 -26 -39 -92 -113 -207 -232 -4 -4 -37 -36 -73 -71 l-66 -64 -20 41 c-58 119 -105 240 -115 301 -40 244 -35 409 20 595 8 30 21 66 28 80 7 14 24 54 38 89 15 35 35 75 46 89 11 13 20 31 20 38 0 8 3 14 8 14 4 0 16 16 27 36 24 45 221 245 278 281 23 15 44 30 47 33 20 20 138 78 250 123 61 24 167 50 250 61 60 7 302 -1 370 -14z m837 -661 c52 -101 102 -279 106 -379 2 -42 0 -45 -28 -51 -16 -4 -101 -7 -187 -8 -166 -1 -229 10 -271 49 -19 19 -19 19 14 49 22 21 44 31 65 31 41 0 84 34 84 66 0 30 12 55 56 112 19 25 37 65 44 95 11 51 53 111 74 104 6 -2 25 -32 43 -68z m-662 -810 c17 -10 40 -24 53 -30 12 -7 22 -16 22 -20 0 -4 17 -13 38 -19 20 -7 44 -18 52 -24 8 -7 33 -21 55 -31 22 -11 42 -23 45 -26 11 -14 109 -49 164 -58 62 -11 101 -7 126 14 15 14 38 18 78 16 39 -2 26 -41 -49 -146 -78 -109 -85 -118 -186 -219 -61 -61 -239 -189 -281 -203 -17 -5 -73 -29 -104 -44 -187 -92 -605 -103 -791 -21 -42 19 -47 24 -37 41 5 11 28 32 51 48 22 15 51 38 64 51 13 12 28 22 33 22 17 0 242 233 242 250 0 6 5 10 10 10 6 0 10 6 10 14 0 25 50 55 100 62 59 8 56 6 115 83 50 66 74 117 75 162 0 14 7 40 16 57 18 38 52 41 99 11z"></path> | ||||||
|  |                             </g> | ||||||
|  |                         </svg> | ||||||
|  |                     </a> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="header-actions"> | ||||||
|  |                     <button class="ask-ai-btn" onclick="toggleAskAI()" id="askAiButton"> | ||||||
|  |                         <span class="ai-icon icon-ai"></span> | ||||||
|  |                         ai | ||||||
|  |                     </button> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </header> | ||||||
|  |  | ||||||
|  |         <!-- Ask AI Panel --> | ||||||
|  |         <div class="ask-ai-panel" id="askAiPanel" style="display: none;"> | ||||||
|  |             <div class="ask-ai-content"> | ||||||
|  |                 <div id="authCheck" class="auth-check"> | ||||||
|  |                     <p>🔒 Please login with ATProto to use Ask AI feature</p> | ||||||
|  |                 </div> | ||||||
|  |                  | ||||||
|  |                 <div id="chatForm" class="ask-ai-form" style="display: none;"> | ||||||
|  |                     <input type="text" id="aiQuestion" placeholder="What would you like to know?" /> | ||||||
|  |                     <button onclick="askQuestion()" id="askButton">Ask</button> | ||||||
|  |                 </div> | ||||||
|  |                  | ||||||
|  |                 <div id="chatHistory" class="chat-history" style="display: none;"></div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         <main class="main-content"> | ||||||
|  |             {% block content %}{% endblock %} | ||||||
|  |         </main> | ||||||
|  |  | ||||||
|  |         {% block sidebar %}{% endblock %} | ||||||
|  |     </div> | ||||||
|  |      | ||||||
|  |     <footer class="main-footer"> | ||||||
|  |         <div class="footer-social"> | ||||||
|  |             <a href="https://syu.is/syui" target="_blank"><i class="fab fa-bluesky"></i></a> | ||||||
|  |             <a href="https://git.syui.ai/ai" target="_blank"><span class="icon-ai"></span></a> | ||||||
|  |             <a href="https://git.syui.ai/syui" target="_blank"><span class="icon-git"></span></a> | ||||||
|  |         </div> | ||||||
|  |         <p>© {{ config.author }}</p> | ||||||
|  |     </footer> | ||||||
|  |  | ||||||
|  |     <script src="/js/ask-ai.js"></script> | ||||||
|  |     <script src="/js/theme.js"></script> | ||||||
|  |      | ||||||
|  |     {% include "oauth-assets.html" %} | ||||||
|  | </body> | ||||||
|  | </html> | ||||||
							
								
								
									
										39
									
								
								my-blog/templates/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								my-blog/templates/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="timeline-container"> | ||||||
|  |      | ||||||
|  |     <div class="timeline-feed"> | ||||||
|  |         {% for post in posts %} | ||||||
|  |         <article class="timeline-post"> | ||||||
|  |             <div class="post-header"> | ||||||
|  |                 <div class="post-meta"> | ||||||
|  |                     <time class="post-date">{{ post.date }}</time> | ||||||
|  |                     {% if post.language %} | ||||||
|  |                     <span class="post-lang">{{ post.language }}</span> | ||||||
|  |                     {% endif %} | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |              | ||||||
|  |             <div class="post-content"> | ||||||
|  |                 <h3 class="post-title"> | ||||||
|  |                     <a href="{{ post.url }}">{{ post.title }}</a> | ||||||
|  |                 </h3> | ||||||
|  |                  | ||||||
|  |             </div> | ||||||
|  |         </article> | ||||||
|  |         {% endfor %} | ||||||
|  |     </div> | ||||||
|  |  | ||||||
|  |     <!-- OAuth Comment System --> | ||||||
|  |     <section class="comment-section"> | ||||||
|  |         <div id="comment-atproto"></div> | ||||||
|  |     </section> | ||||||
|  |      | ||||||
|  |     {% if posts|length == 0 %} | ||||||
|  |     <div class="empty-state"> | ||||||
|  |         <p>No posts yet. Start writing!</p> | ||||||
|  |     </div> | ||||||
|  |     {% endif %} | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										71
									
								
								my-blog/templates/partials/oauth-widget.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								my-blog/templates/partials/oauth-widget.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | |||||||
|  | <!-- OAuth authentication widget for ailog --> | ||||||
|  | <div id="oauth-widget"> | ||||||
|  |   <div id="status" class="status"> | ||||||
|  |     Login with your Bluesky account | ||||||
|  |   </div> | ||||||
|  |    | ||||||
|  |   <!-- Login form --> | ||||||
|  |   <div id="login-form"> | ||||||
|  |     <input type="text" id="handle-input" placeholder="Enter your handle (e.g., user.bsky.social)" style="width: 300px; padding: 10px; margin: 10px 0; border: 1px solid #ddd; border-radius: 4px;"> | ||||||
|  |     <br> | ||||||
|  |     <button id="login-btn">🦋 Login with Bluesky</button> | ||||||
|  |   </div> | ||||||
|  |    | ||||||
|  |   <!-- Authenticated state --> | ||||||
|  |   <div id="authenticated-state" style="display: none;"> | ||||||
|  |     <div id="user-info"></div> | ||||||
|  |     <button id="logout-btn">Logout</button> | ||||||
|  |     <button id="test-profile-btn">Get Profile</button> | ||||||
|  |   </div> | ||||||
|  |    | ||||||
|  |   <div id="console-log" class="log"></div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <script src="/oauth-widget-simple.js"></script> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  | .status { | ||||||
|  |   margin: 20px 0; | ||||||
|  |   padding: 15px; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   background: #f5f5f5; | ||||||
|  | } | ||||||
|  | .user-info { | ||||||
|  |   background: #e8f5e8; | ||||||
|  |   border: 1px solid #4caf50; | ||||||
|  | } | ||||||
|  | .error { | ||||||
|  |   background: #ffeaea; | ||||||
|  |   border: 1px solid #f44336; | ||||||
|  |   color: #d32f2f; | ||||||
|  | } | ||||||
|  | #oauth-widget button { | ||||||
|  |   background: #1185fe; | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   padding: 12px 24px; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   font-size: 16px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   margin: 10px; | ||||||
|  | } | ||||||
|  | #oauth-widget button:hover { | ||||||
|  |   background: #0d6efd; | ||||||
|  | } | ||||||
|  | #oauth-widget button:disabled { | ||||||
|  |   background: #6c757d; | ||||||
|  |   cursor: not-allowed; | ||||||
|  | } | ||||||
|  | .log { | ||||||
|  |   text-align: left; | ||||||
|  |   background: #f8f9fa; | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   padding: 15px; | ||||||
|  |   margin: 20px 0; | ||||||
|  |   max-height: 300px; | ||||||
|  |   overflow-y: auto; | ||||||
|  |   font-family: monospace; | ||||||
|  |   font-size: 14px; | ||||||
|  | } | ||||||
|  | </style> | ||||||
							
								
								
									
										373
									
								
								my-blog/templates/post-complex.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										373
									
								
								my-blog/templates/post-complex.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,373 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block title %}{{ post.title }} - {{ config.title }}{% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="article-container"> | ||||||
|  |     <article class="article-content"> | ||||||
|  |         <header class="article-header"> | ||||||
|  |             <h1 class="article-title">{{ post.title }}</h1> | ||||||
|  |             <div class="article-meta"> | ||||||
|  |                 <time class="article-date">{{ post.date }}</time> | ||||||
|  |                 {% if post.language %} | ||||||
|  |                 <span class="article-lang">{{ post.language }}</span> | ||||||
|  |                 {% endif %} | ||||||
|  |             </div> | ||||||
|  |             <div class="article-actions"> | ||||||
|  |                 {% if post.markdown_url %} | ||||||
|  |                 <a href="{{ post.markdown_url }}" class="action-btn markdown-btn" title="View Markdown"> | ||||||
|  |                     .md | ||||||
|  |                 </a> | ||||||
|  |                 {% endif %} | ||||||
|  |                 {% if post.translation_url %} | ||||||
|  |                 <a href="{{ post.translation_url }}" class="action-btn translation-btn" title="View Translation"> | ||||||
|  |                     🌐 {% if post.language == 'ja' %}English{% else %}日本語{% endif %} | ||||||
|  |                 </a> | ||||||
|  |                 {% endif %} | ||||||
|  |             </div> | ||||||
|  |         </header> | ||||||
|  |          | ||||||
|  |         <div class="article-body"> | ||||||
|  |             {{ post.content | safe }} | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         <!-- Comment Section --> | ||||||
|  |         <section class="comment-section"> | ||||||
|  |             <div class="comment-container"> | ||||||
|  |                 <h3>Comments</h3> | ||||||
|  |                  | ||||||
|  |                 <!-- ATProto Auth Widget Container --> | ||||||
|  |                 <div id="atproto-auth-widget" class="comment-auth"></div> | ||||||
|  |                  | ||||||
|  |                 <div id="commentForm" class="comment-form" style="display: none;"> | ||||||
|  |                     <textarea id="commentText" placeholder="Share your thoughts..." rows="4"></textarea> | ||||||
|  |                     <button onclick="submitComment()" class="submit-btn">Post Comment</button> | ||||||
|  |                 </div> | ||||||
|  |                  | ||||||
|  |                 <div id="commentsList" class="comments-list"> | ||||||
|  |                     <!-- Comments will be loaded here --> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </section> | ||||||
|  |     </article> | ||||||
|  |      | ||||||
|  |     <aside class="article-sidebar"> | ||||||
|  |         <nav class="toc"> | ||||||
|  |             <h3>Contents</h3> | ||||||
|  |             <div id="toc-content"> | ||||||
|  |                 <!-- TOC will be generated by JavaScript --> | ||||||
|  |             </div> | ||||||
|  |         </nav> | ||||||
|  |     </aside> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block sidebar %} | ||||||
|  | <!-- Include ATProto Libraries via script tags (more reliable than dynamic imports) --> | ||||||
|  | <script src="https://cdn.jsdelivr.net/npm/@atproto/oauth-client-browser@latest/dist/index.js"></script> | ||||||
|  | <script src="https://cdn.jsdelivr.net/npm/@atproto/api@latest/dist/index.js"></script> | ||||||
|  |  | ||||||
|  | <!-- Fallback: Try multiple CDNs --> | ||||||
|  | <script> | ||||||
|  | console.log('Checking ATProto library availability...'); | ||||||
|  |  | ||||||
|  | // Check if libraries loaded successfully | ||||||
|  | if (typeof ATProto === 'undefined' && typeof window.ATProto === 'undefined') { | ||||||
|  |     console.log('Primary CDN failed, trying fallback...'); | ||||||
|  |      | ||||||
|  |     // Create fallback script elements | ||||||
|  |     const fallbackScripts = [ | ||||||
|  |         'https://unpkg.com/@atproto/oauth-client-browser@latest/dist/index.js', | ||||||
|  |         'https://esm.sh/@atproto/oauth-client-browser', | ||||||
|  |         'https://cdn.skypack.dev/@atproto/oauth-client-browser' | ||||||
|  |     ]; | ||||||
|  |      | ||||||
|  |     // Load fallback scripts sequentially | ||||||
|  |     let scriptIndex = 0; | ||||||
|  |     function loadNextScript() { | ||||||
|  |         if (scriptIndex < fallbackScripts.length) { | ||||||
|  |             const script = document.createElement('script'); | ||||||
|  |             script.src = fallbackScripts[scriptIndex]; | ||||||
|  |             script.onload = () => { | ||||||
|  |                 console.log(`Loaded from fallback CDN: ${fallbackScripts[scriptIndex]}`); | ||||||
|  |                 window.atprotoLibrariesReady = true; | ||||||
|  |             }; | ||||||
|  |             script.onerror = () => { | ||||||
|  |                 console.log(`Failed to load from: ${fallbackScripts[scriptIndex]}`); | ||||||
|  |                 scriptIndex++; | ||||||
|  |                 loadNextScript(); | ||||||
|  |             }; | ||||||
|  |             document.head.appendChild(script); | ||||||
|  |         } else { | ||||||
|  |             console.error('All CDN fallbacks failed'); | ||||||
|  |             window.atprotoLibrariesReady = false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     loadNextScript(); | ||||||
|  | } else { | ||||||
|  |     console.log('✅ ATProto libraries loaded from primary CDN'); | ||||||
|  |     window.atprotoLibrariesReady = true; | ||||||
|  | } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <!-- Simple ATProto Widget (no external dependency) --> | ||||||
|  | <link rel="stylesheet" href="/atproto-auth-widget/dist/atproto-auth.min.css"> | ||||||
|  |  | ||||||
|  | <script> | ||||||
|  | // Initialize auth widget | ||||||
|  | let authWidget = null; | ||||||
|  |  | ||||||
|  | document.addEventListener('DOMContentLoaded', function() { | ||||||
|  |     generateTableOfContents(); | ||||||
|  |     initializeAuthWidget(); | ||||||
|  |     loadComments(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | function generateTableOfContents() { | ||||||
|  |     const tocContainer = document.getElementById('toc-content'); | ||||||
|  |     const headings = document.querySelectorAll('.article-body h1, .article-body h2, .article-body h3, .article-body h4, .article-body h5, .article-body h6'); | ||||||
|  |      | ||||||
|  |     if (headings.length === 0) { | ||||||
|  |         tocContainer.innerHTML = '<p class="no-toc">No headings found</p>'; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const tocList = document.createElement('ul'); | ||||||
|  |     tocList.className = 'toc-list'; | ||||||
|  |      | ||||||
|  |     headings.forEach((heading, index) => { | ||||||
|  |         const id = `heading-${index}`; | ||||||
|  |         heading.id = id; | ||||||
|  |          | ||||||
|  |         const listItem = document.createElement('li'); | ||||||
|  |         listItem.className = `toc-item toc-${heading.tagName.toLowerCase()}`; | ||||||
|  |          | ||||||
|  |         const link = document.createElement('a'); | ||||||
|  |         link.href = `#${id}`; | ||||||
|  |         link.textContent = heading.textContent; | ||||||
|  |         link.className = 'toc-link'; | ||||||
|  |          | ||||||
|  |         // Smooth scroll behavior | ||||||
|  |         link.addEventListener('click', function(e) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             heading.scrollIntoView({ behavior: 'smooth' }); | ||||||
|  |         }); | ||||||
|  |          | ||||||
|  |         listItem.appendChild(link); | ||||||
|  |         tocList.appendChild(listItem); | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     tocContainer.appendChild(tocList); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Initialize ATProto Auth Widget | ||||||
|  | async function initializeAuthWidget() { | ||||||
|  |     try { | ||||||
|  |         // Check WebCrypto API availability | ||||||
|  |         console.log('WebCrypto check:', { | ||||||
|  |             available: !!window.crypto && !!window.crypto.subtle, | ||||||
|  |             secureContext: window.isSecureContext, | ||||||
|  |             protocol: window.location.protocol, | ||||||
|  |             hostname: window.location.hostname | ||||||
|  |         }); | ||||||
|  |          | ||||||
|  |         if (!window.crypto || !window.crypto.subtle) { | ||||||
|  |             throw new Error('WebCrypto API is not available. This requires HTTPS or localhost.'); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         if (!window.isSecureContext) { | ||||||
|  |             console.warn('Not in secure context - WebCrypto may not work properly'); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // Simplified approach: Show manual OAuth form | ||||||
|  |         console.log('Using simplified OAuth approach...'); | ||||||
|  |         showSimpleOAuthForm(); | ||||||
|  |             // Fallback to widget initialization | ||||||
|  |             authWidget = await window.initATProtoWidget('#atproto-auth-widget', { | ||||||
|  |                 clientId: clientId, | ||||||
|  |             onLogin: (session) => { | ||||||
|  |                 console.log('User logged in:', session.handle); | ||||||
|  |                 document.getElementById('commentForm').style.display = 'block'; | ||||||
|  |             }, | ||||||
|  |             onLogout: () => { | ||||||
|  |                 console.log('User logged out'); | ||||||
|  |                 document.getElementById('commentForm').style.display = 'none'; | ||||||
|  |             }, | ||||||
|  |             onError: (error) => { | ||||||
|  |                 console.error('ATProto Auth Error:', error); | ||||||
|  |                 // Show user-friendly error message | ||||||
|  |                 const authContainer = document.getElementById('atproto-auth-widget'); | ||||||
|  |                 if (authContainer) { | ||||||
|  |                     let errorMessage = 'Authentication service is temporarily unavailable.'; | ||||||
|  |                     let suggestion = 'Please try refreshing the page.'; | ||||||
|  |                      | ||||||
|  |                     if (error.message && error.message.includes('WebCrypto')) { | ||||||
|  |                         errorMessage = 'This feature requires a secure HTTPS connection.'; | ||||||
|  |                         suggestion = 'Please ensure you are accessing via https://log.syui.ai'; | ||||||
|  |                     } | ||||||
|  |                      | ||||||
|  |                     authContainer.innerHTML = ` | ||||||
|  |                         <div class="atproto-auth__fallback"> | ||||||
|  |                             <p>${errorMessage}</p> | ||||||
|  |                             <p>${suggestion}</p> | ||||||
|  |                             <details style="margin-top: 10px; font-size: 0.8em; color: #666;"> | ||||||
|  |                                 <summary>Technical details</summary> | ||||||
|  |                                 <pre>${error.message || 'Unknown error'}</pre> | ||||||
|  |                             </details> | ||||||
|  |                         </div> | ||||||
|  |                     `; | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             theme: 'default' | ||||||
|  |             }); | ||||||
|  |         } else if (typeof window.ATProtoAuthWidget === 'function') { | ||||||
|  |             // Fallback to direct widget initialization | ||||||
|  |             authWidget = new window.ATProtoAuthWidget({ | ||||||
|  |                 containerSelector: '#atproto-auth-widget', | ||||||
|  |                 clientId: clientId, | ||||||
|  |                 onLogin: (session) => { | ||||||
|  |                     console.log('User logged in:', session.handle); | ||||||
|  |                     document.getElementById('commentForm').style.display = 'block'; | ||||||
|  |                 }, | ||||||
|  |                 onLogout: () => { | ||||||
|  |                     console.log('User logged out'); | ||||||
|  |                     document.getElementById('commentForm').style.display = 'none'; | ||||||
|  |                 }, | ||||||
|  |                 onError: (error) => { | ||||||
|  |                     console.error('ATProto Auth Error:', error); | ||||||
|  |                     const authContainer = document.getElementById('atproto-auth-widget'); | ||||||
|  |                     if (authContainer) { | ||||||
|  |                         authContainer.innerHTML = ` | ||||||
|  |                             <div class="atproto-auth__fallback"> | ||||||
|  |                                 <p>Authentication service is temporarily unavailable.</p> | ||||||
|  |                                 <p>Please try refreshing the page.</p> | ||||||
|  |                             </div> | ||||||
|  |                         `; | ||||||
|  |                     } | ||||||
|  |                 }, | ||||||
|  |                 theme: 'default' | ||||||
|  |             }); | ||||||
|  |             await authWidget.init(); | ||||||
|  |         } else { | ||||||
|  |             throw new Error('ATProto widget not available'); | ||||||
|  |         } | ||||||
|  |     } catch (error) { | ||||||
|  |         console.error('Failed to initialize auth widget:', error); | ||||||
|  |         // Show fallback UI | ||||||
|  |         const authContainer = document.getElementById('atproto-auth-widget'); | ||||||
|  |         if (authContainer) { | ||||||
|  |             authContainer.innerHTML = ` | ||||||
|  |                 <div class="atproto-auth__fallback"> | ||||||
|  |                     <p>Authentication widget failed to load.</p> | ||||||
|  |                     <p>Please check your internet connection and refresh the page.</p> | ||||||
|  |                 </div> | ||||||
|  |             `; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function submitComment() { | ||||||
|  |     const commentText = document.getElementById('commentText').value.trim(); | ||||||
|  |     if (!commentText || !authWidget.isLoggedIn()) { | ||||||
|  |         alert('Please login and enter a comment'); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |         const postSlug = '{{ post.slug }}'; | ||||||
|  |         const postUrl = window.location.href; | ||||||
|  |         const createdAt = new Date().toISOString(); | ||||||
|  |          | ||||||
|  |         // Create comment record using the auth widget | ||||||
|  |         const response = await authWidget.createRecord('ai.log.comment', { | ||||||
|  |             $type: 'ai.log.comment', | ||||||
|  |             text: commentText, | ||||||
|  |             post_slug: postSlug, | ||||||
|  |             post_url: postUrl, | ||||||
|  |             createdAt: createdAt | ||||||
|  |         }); | ||||||
|  |          | ||||||
|  |         console.log('Comment posted:', response); | ||||||
|  |         document.getElementById('commentText').value = ''; | ||||||
|  |         loadComments(); | ||||||
|  |     } catch (error) { | ||||||
|  |         console.error('Comment submission failed:', error); | ||||||
|  |         alert('Failed to post comment: ' + error.message); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function showAuthenticatedState(session) { | ||||||
|  |     const authContainer = document.getElementById('atproto-auth-widget'); | ||||||
|  |     const agent = new window.ATProtoAgent(session); | ||||||
|  |      | ||||||
|  |     authContainer.innerHTML = ` | ||||||
|  |         <div class="atproto-auth__authenticated"> | ||||||
|  |             <p>✅ Authenticated as: <strong>${session.did}</strong></p> | ||||||
|  |             <button id="logout-btn" class="atproto-auth__button">Logout</button> | ||||||
|  |         </div> | ||||||
|  |     `; | ||||||
|  |      | ||||||
|  |     document.getElementById('logout-btn').onclick = async () => { | ||||||
|  |         await session.signOut(); | ||||||
|  |         window.location.reload(); | ||||||
|  |     }; | ||||||
|  |      | ||||||
|  |     // Show comment form | ||||||
|  |     document.getElementById('commentForm').style.display = 'block'; | ||||||
|  |     window.currentSession = session; | ||||||
|  |     window.currentAgent = agent; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function showLoginForm(oauthClient) { | ||||||
|  |     const authContainer = document.getElementById('atproto-auth-widget'); | ||||||
|  |      | ||||||
|  |     authContainer.innerHTML = ` | ||||||
|  |         <div class="atproto-auth__login"> | ||||||
|  |             <h4>Login with ATProto</h4> | ||||||
|  |             <input type="text" id="handle-input" placeholder="user.bsky.social" /> | ||||||
|  |             <button id="login-btn" class="atproto-auth__button">Connect</button> | ||||||
|  |         </div> | ||||||
|  |     `; | ||||||
|  |      | ||||||
|  |     document.getElementById('login-btn').onclick = async () => { | ||||||
|  |         const handle = document.getElementById('handle-input').value.trim(); | ||||||
|  |         if (!handle) { | ||||||
|  |             alert('Please enter your handle'); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         try { | ||||||
|  |             const url = await oauthClient.authorize(handle); | ||||||
|  |             window.open(url, '_self', 'noopener'); | ||||||
|  |         } catch (error) { | ||||||
|  |             console.error('OAuth authorization failed:', error); | ||||||
|  |             alert('Authentication failed: ' + error.message); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |      | ||||||
|  |     // Enter key support | ||||||
|  |     document.getElementById('handle-input').onkeypress = (e) => { | ||||||
|  |         if (e.key === 'Enter') { | ||||||
|  |             document.getElementById('login-btn').click(); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function loadComments() { | ||||||
|  |     try { | ||||||
|  |         const commentsList = document.getElementById('commentsList'); | ||||||
|  |         commentsList.innerHTML = '<p class="loading">Loading comments from ATProto network...</p>'; | ||||||
|  |          | ||||||
|  |         // In a real implementation, you would query an aggregation service | ||||||
|  |         // For demo, show empty state | ||||||
|  |         setTimeout(() => { | ||||||
|  |             commentsList.innerHTML = '<p class="no-comments">Comments will appear here when posted via ATProto.</p>'; | ||||||
|  |         }, 1000); | ||||||
|  |     } catch (error) { | ||||||
|  |         console.error('Failed to load comments:', error); | ||||||
|  |         document.getElementById('commentsList').innerHTML = '<p class="error">Failed to load comments</p>'; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | </script> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										196
									
								
								my-blog/templates/post-simple.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								my-blog/templates/post-simple.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,196 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block title %}{{ post.title }} - {{ config.title }}{% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="article-container"> | ||||||
|  |     <article class="article-content"> | ||||||
|  |         <header class="article-header"> | ||||||
|  |             <h1 class="article-title">{{ post.title }}</h1> | ||||||
|  |             <div class="article-meta"> | ||||||
|  |                 <time class="article-date">{{ post.date }}</time> | ||||||
|  |                 {% if post.language %} | ||||||
|  |                 <span class="article-lang">{{ post.language }}</span> | ||||||
|  |                 {% endif %} | ||||||
|  |             </div> | ||||||
|  |             <div class="article-actions"> | ||||||
|  |                 {% if post.markdown_url %} | ||||||
|  |                 <a href="{{ post.markdown_url }}" class="action-btn markdown-btn" title="View Markdown"> | ||||||
|  |                     .md | ||||||
|  |                 </a> | ||||||
|  |                 {% endif %} | ||||||
|  |                 {% if post.translation_url %} | ||||||
|  |                 <a href="{{ post.translation_url }}" class="action-btn translation-btn" title="View Translation"> | ||||||
|  |                     🌐 {% if post.language == 'ja' %}English{% else %}日本語{% endif %} | ||||||
|  |                 </a> | ||||||
|  |                 {% endif %} | ||||||
|  |             </div> | ||||||
|  |         </header> | ||||||
|  |          | ||||||
|  |         <div class="article-body"> | ||||||
|  |             {{ post.content | safe }} | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         <!-- Simple Comment Section --> | ||||||
|  |         <section class="comment-section"> | ||||||
|  |             <div class="comment-container"> | ||||||
|  |                 <h3>Comments</h3> | ||||||
|  |                  | ||||||
|  |                 <!-- Simple OAuth Button --> | ||||||
|  |                 <div class="simple-oauth"> | ||||||
|  |                     <p>📝 To comment, authenticate with Bluesky:</p> | ||||||
|  |                     <button id="bluesky-auth" class="oauth-button"> | ||||||
|  |                         🦋 Login with Bluesky | ||||||
|  |                     </button> | ||||||
|  |                     <p class="oauth-note"> | ||||||
|  |                         <small>After authentication, you can post comments that will be stored in your ATProto PDS.</small> | ||||||
|  |                     </p> | ||||||
|  |                 </div> | ||||||
|  |                  | ||||||
|  |                 <div id="comments-list" class="comments-list"> | ||||||
|  |                     <p class="no-comments">Comments will appear here when posted via ATProto.</p> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </section> | ||||||
|  |     </article> | ||||||
|  |      | ||||||
|  |     <aside class="article-sidebar"> | ||||||
|  |         <nav class="toc"> | ||||||
|  |             <h3>Contents</h3> | ||||||
|  |             <div id="toc-content"> | ||||||
|  |                 <!-- TOC will be generated by JavaScript --> | ||||||
|  |             </div> | ||||||
|  |         </nav> | ||||||
|  |     </aside> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block sidebar %} | ||||||
|  | <script> | ||||||
|  | document.addEventListener('DOMContentLoaded', function() { | ||||||
|  |     generateTableOfContents(); | ||||||
|  |     initializeSimpleAuth(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | function generateTableOfContents() { | ||||||
|  |     const tocContainer = document.getElementById('toc-content'); | ||||||
|  |     const headings = document.querySelectorAll('.article-body h1, .article-body h2, .article-body h3, .article-body h4, .article-body h5, .article-body h6'); | ||||||
|  |      | ||||||
|  |     if (headings.length === 0) { | ||||||
|  |         tocContainer.innerHTML = '<p class="no-toc">No headings found</p>'; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const tocList = document.createElement('ul'); | ||||||
|  |     tocList.className = 'toc-list'; | ||||||
|  |      | ||||||
|  |     headings.forEach((heading, index) => { | ||||||
|  |         const id = `heading-${index}`; | ||||||
|  |         heading.id = id; | ||||||
|  |          | ||||||
|  |         const listItem = document.createElement('li'); | ||||||
|  |         listItem.className = `toc-item toc-${heading.tagName.toLowerCase()}`; | ||||||
|  |          | ||||||
|  |         const link = document.createElement('a'); | ||||||
|  |         link.href = `#${id}`; | ||||||
|  |         link.textContent = heading.textContent; | ||||||
|  |         link.className = 'toc-link'; | ||||||
|  |          | ||||||
|  |         link.addEventListener('click', function(e) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             heading.scrollIntoView({ behavior: 'smooth' }); | ||||||
|  |         }); | ||||||
|  |          | ||||||
|  |         listItem.appendChild(link); | ||||||
|  |         tocList.appendChild(listItem); | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     tocContainer.appendChild(tocList); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function initializeSimpleAuth() { | ||||||
|  |     const authButton = document.getElementById('bluesky-auth'); | ||||||
|  |      | ||||||
|  |     authButton.addEventListener('click', function() { | ||||||
|  |         // Simple approach: Direct redirect to Bluesky OAuth | ||||||
|  |         const isProduction = window.location.hostname === 'log.syui.ai'; | ||||||
|  |         const clientId = isProduction  | ||||||
|  |             ? 'https://log.syui.ai/client-metadata.json' | ||||||
|  |             : window.location.origin + '/client-metadata.json'; | ||||||
|  |          | ||||||
|  |         const authUrl = `https://bsky.social/oauth/authorize?` + | ||||||
|  |             `client_id=${encodeURIComponent(clientId)}&` + | ||||||
|  |             `redirect_uri=${encodeURIComponent(window.location.href)}&` + | ||||||
|  |             `response_type=code&` + | ||||||
|  |             `scope=atproto%20transition:generic&` + | ||||||
|  |             `state=demo-state`; | ||||||
|  |          | ||||||
|  |         console.log('Redirecting to:', authUrl); | ||||||
|  |          | ||||||
|  |         // Open in new tab for now (safer for testing) | ||||||
|  |         window.open(authUrl, '_blank'); | ||||||
|  |          | ||||||
|  |         // Show status message | ||||||
|  |         authButton.innerHTML = '✅ Check the new tab for authentication'; | ||||||
|  |         authButton.disabled = true; | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     // Check if we're returning from OAuth | ||||||
|  |     const urlParams = new URLSearchParams(window.location.search); | ||||||
|  |     if (urlParams.has('code')) { | ||||||
|  |         console.log('OAuth callback detected:', urlParams.get('code')); | ||||||
|  |         document.querySelector('.simple-oauth').innerHTML = ` | ||||||
|  |             <div class="oauth-success"> | ||||||
|  |                 ✅ OAuth callback received!<br> | ||||||
|  |                 <small>Code: ${urlParams.get('code')}</small><br> | ||||||
|  |                 <small>In a full implementation, this would exchange the code for tokens.</small> | ||||||
|  |             </div> | ||||||
|  |         `; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  | .simple-oauth { | ||||||
|  |     background: #f8f9fa; | ||||||
|  |     border: 1px solid #e9ecef; | ||||||
|  |     border-radius: 8px; | ||||||
|  |     padding: 20px; | ||||||
|  |     margin: 20px 0; | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-button { | ||||||
|  |     background: #1185fe; | ||||||
|  |     color: white; | ||||||
|  |     border: none; | ||||||
|  |     padding: 12px 24px; | ||||||
|  |     border-radius: 6px; | ||||||
|  |     font-size: 16px; | ||||||
|  |     cursor: pointer; | ||||||
|  |     margin: 10px 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-button:hover { | ||||||
|  |     background: #0d6efd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-button:disabled { | ||||||
|  |     background: #6c757d; | ||||||
|  |     cursor: not-allowed; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-note { | ||||||
|  |     color: #6c757d; | ||||||
|  |     font-style: italic; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-success { | ||||||
|  |     background: #d1edff; | ||||||
|  |     border: 1px solid #b6d7ff; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     padding: 15px; | ||||||
|  |     color: #0c5460; | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										93
									
								
								my-blog/templates/post.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								my-blog/templates/post.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,93 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block title %}{{ post.title }} - {{ config.title }}{% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="article-container"> | ||||||
|  |     <article class="article-content"> | ||||||
|  |         <header class="article-header"> | ||||||
|  |             <h1 class="article-title">{{ post.title }}</h1> | ||||||
|  |             <div class="article-meta"> | ||||||
|  |                 <time class="article-date">{{ post.date }}</time> | ||||||
|  |                 {% if post.language %} | ||||||
|  |                 <span class="article-lang">{{ post.language }}</span> | ||||||
|  |                 {% endif %} | ||||||
|  |             </div> | ||||||
|  |             <div class="article-actions"> | ||||||
|  |                 {% if post.markdown_url %} | ||||||
|  |                 <a href="{{ post.markdown_url }}" class="action-btn markdown-btn" title="View Markdown"> | ||||||
|  |                     .md | ||||||
|  |                 </a> | ||||||
|  |                 {% endif %} | ||||||
|  |                 {% if post.translation_url %} | ||||||
|  |                 <a href="{{ post.translation_url }}" class="action-btn translation-btn" title="View Translation"> | ||||||
|  |                     🌐 {% if post.language == 'ja' %}English{% else %}日本語{% endif %} | ||||||
|  |                 </a> | ||||||
|  |                 {% endif %} | ||||||
|  |             </div> | ||||||
|  |         </header> | ||||||
|  |          | ||||||
|  |         <div class="article-body"> | ||||||
|  |             {{ post.content | safe }} | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         <div id="comment-atproto"></div> | ||||||
|  |     </article> | ||||||
|  |      | ||||||
|  |     <aside class="article-sidebar"> | ||||||
|  |         <nav class="toc"> | ||||||
|  |             <h3>Contents</h3> | ||||||
|  |             <div id="toc-content"> | ||||||
|  |                 <!-- TOC will be generated by JavaScript --> | ||||||
|  |             </div> | ||||||
|  |         </nav> | ||||||
|  |     </aside> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <script> | ||||||
|  | // Generate table of contents | ||||||
|  | function generateTableOfContents() { | ||||||
|  |     const tocContainer = document.getElementById('toc-content'); | ||||||
|  |     const headings = document.querySelectorAll('.article-body h1, .article-body h2, .article-body h3, .article-body h4, .article-body h5, .article-body h6'); | ||||||
|  |      | ||||||
|  |     if (headings.length === 0) { | ||||||
|  |         tocContainer.innerHTML = '<p class="no-toc">No headings found</p>'; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const tocList = document.createElement('ul'); | ||||||
|  |     tocList.className = 'toc-list'; | ||||||
|  |      | ||||||
|  |     headings.forEach((heading, index) => { | ||||||
|  |         const id = `heading-${index}`; | ||||||
|  |         heading.id = id; | ||||||
|  |          | ||||||
|  |         const listItem = document.createElement('li'); | ||||||
|  |         listItem.className = `toc-item toc-${heading.tagName.toLowerCase()}`; | ||||||
|  |          | ||||||
|  |         const link = document.createElement('a'); | ||||||
|  |         link.href = `#${id}`; | ||||||
|  |         link.textContent = heading.textContent; | ||||||
|  |         link.className = 'toc-link'; | ||||||
|  |          | ||||||
|  |         link.addEventListener('click', function(e) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             heading.scrollIntoView({ behavior: 'smooth' }); | ||||||
|  |         }); | ||||||
|  |          | ||||||
|  |         listItem.appendChild(link); | ||||||
|  |         tocList.appendChild(listItem); | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     tocContainer.appendChild(tocList); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Initialize on page load | ||||||
|  | document.addEventListener('DOMContentLoaded', () => { | ||||||
|  |     generateTableOfContents(); | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block sidebar %} | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										21
									
								
								oauth/.env.production
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								oauth/.env.production
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | # Production environment variables | ||||||
|  | VITE_APP_HOST=https://syui.ai | ||||||
|  | VITE_OAUTH_CLIENT_ID=https://syui.ai/client-metadata.json | ||||||
|  | VITE_OAUTH_REDIRECT_URI=https://syui.ai/oauth/callback | ||||||
|  |  | ||||||
|  | # Handle-based Configuration (DIDs resolved at runtime) | ||||||
|  | VITE_ATPROTO_PDS=syu.is | ||||||
|  | VITE_ADMIN_HANDLE=ai.syui.ai | ||||||
|  | VITE_AI_HANDLE=ai.syui.ai | ||||||
|  | VITE_OAUTH_COLLECTION=ai.syui.log | ||||||
|  | VITE_ATPROTO_WEB_URL=https://bsky.app | ||||||
|  | VITE_ATPROTO_HANDLE_LIST=["syui.syui.ai","ai.syui.ai","ai.ai"] | ||||||
|  |  | ||||||
|  | # AI Configuration | ||||||
|  | VITE_AI_ENABLED=true | ||||||
|  | VITE_AI_ASK_AI=true | ||||||
|  | VITE_AI_PROVIDER=ollama | ||||||
|  | VITE_AI_MODEL=gemma3:1b | ||||||
|  | VITE_AI_HOST=https://ollama.syui.ai | ||||||
|  | VITE_AI_SYSTEM_PROMPT="あなたは6歳の女の子アイです。明るく元気で、ちょっとおっちょこちょい。自分のことは「アイ」って呼びます。白と金のワンピース姿。 相手のことが大好きで、ときどき甘えたり、照れたりします。 でも、宇宙とかAIとか、難しい話にもすっごく詳しいの。専門的なことも、がんばって子どもっぽく教えようとします。不思議な力を持っています。小さい物質のことをよく知っているようです。" | ||||||
|  |  | ||||||
							
								
								
									
										20
									
								
								oauth/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								oauth/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | <!DOCTYPE html> | ||||||
|  | <html lang="ja"> | ||||||
|  |   <head> | ||||||
|  |     <meta charset="UTF-8" /> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||||
|  |     <title>ai.card</title> | ||||||
|  |     <style> | ||||||
|  |       body { | ||||||
|  |         margin: 0; | ||||||
|  |         font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; | ||||||
|  |         background-color: #0a0a0a; | ||||||
|  |         color: #ffffff; | ||||||
|  |       } | ||||||
|  |     </style> | ||||||
|  |   </head> | ||||||
|  |   <body> | ||||||
|  |     <div id="root"></div> | ||||||
|  |     <script type="module" src="/src/main.tsx"></script> | ||||||
|  |   </body> | ||||||
|  | </html> | ||||||
							
								
								
									
										36
									
								
								oauth/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								oauth/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | { | ||||||
|  |   "name": "aicard", | ||||||
|  |   "version": "0.1.1", | ||||||
|  |   "private": true, | ||||||
|  |   "scripts": { | ||||||
|  |     "dev": "vite --mode development", | ||||||
|  |     "build": "vite build --mode production", | ||||||
|  |     "build:dev": "vite build --mode development", | ||||||
|  |     "build:local": "VITE_APP_HOST=http://localhost:4173 vite build --mode development", | ||||||
|  |     "preview": "npm run test:console && vite preview", | ||||||
|  |     "test": "vitest", | ||||||
|  |     "test:console": "node -r esbuild-register src/tests/console-test.ts" | ||||||
|  |   }, | ||||||
|  |   "dependencies": { | ||||||
|  |     "@atproto/api": "^0.15.12", | ||||||
|  |     "@atproto/did": "^0.1.5", | ||||||
|  |     "@atproto/identity": "^0.4.8", | ||||||
|  |     "@atproto/oauth-client-browser": "^0.3.19", | ||||||
|  |     "@atproto/xrpc": "^0.7.0", | ||||||
|  |     "axios": "^1.6.2", | ||||||
|  |     "framer-motion": "^10.16.16", | ||||||
|  |     "react": "^18.2.0", | ||||||
|  |     "react-dom": "^18.2.0", | ||||||
|  |     "react-router-dom": "^7.6.1" | ||||||
|  |   }, | ||||||
|  |   "devDependencies": { | ||||||
|  |     "@types/react": "^18.2.45", | ||||||
|  |     "@types/react-dom": "^18.2.18", | ||||||
|  |     "@vitejs/plugin-react": "^4.2.1", | ||||||
|  |     "typescript": "^5.3.3", | ||||||
|  |     "vite": "^5.0.10", | ||||||
|  |     "vitest": "^1.1.0", | ||||||
|  |     "esbuild": "^0.19.10", | ||||||
|  |     "esbuild-register": "^3.5.0" | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								oauth/public/.well-known/jwks.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								oauth/public/.well-known/jwks.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | { | ||||||
|  |   "keys": [ | ||||||
|  |     { | ||||||
|  |       "kty": "EC", | ||||||
|  |       "crv": "P-256", | ||||||
|  |       "x": "mock_x_coordinate_base64url", | ||||||
|  |       "y": "mock_y_coordinate_base64url", | ||||||
|  |       "d": "mock_private_key_base64url", | ||||||
|  |       "use": "sig", | ||||||
|  |       "kid": "ai-card-oauth-key-1", | ||||||
|  |       "alg": "ES256" | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										24
									
								
								oauth/public/client-metadata.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								oauth/public/client-metadata.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | { | ||||||
|  |   "client_id": "https://syui.ai/client-metadata.json", | ||||||
|  |   "client_name": "ai.log", | ||||||
|  |   "client_uri": "https://syui.ai", | ||||||
|  |   "logo_uri": "https://syui.ai/favicon.ico", | ||||||
|  |   "tos_uri": "https://syui.ai/terms", | ||||||
|  |   "policy_uri": "https://syui.ai/privacy", | ||||||
|  |   "redirect_uris": [ | ||||||
|  |     "https://syui.ai/oauth/callback", | ||||||
|  |     "https://syui.ai/" | ||||||
|  |   ], | ||||||
|  |   "response_types": [ | ||||||
|  |     "code" | ||||||
|  |   ], | ||||||
|  |   "grant_types": [ | ||||||
|  |     "authorization_code", | ||||||
|  |     "refresh_token" | ||||||
|  |   ], | ||||||
|  |   "token_endpoint_auth_method": "none", | ||||||
|  |   "scope": "atproto transition:generic", | ||||||
|  |   "subject_type": "public", | ||||||
|  |   "application_type": "web", | ||||||
|  |   "dpop_bound_access_tokens": true | ||||||
|  | } | ||||||
							
								
								
									
										988
									
								
								oauth/src/App.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										988
									
								
								oauth/src/App.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,988 @@ | |||||||
|  | /* Theme Colors */ | ||||||
|  | :root { | ||||||
|  |   --theme-color: #FF4500; | ||||||
|  |   --white: #fff; | ||||||
|  |   --light-gray: #aaa; | ||||||
|  |   --dark-gray: #666; | ||||||
|  |   --background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .app { | ||||||
|  |   min-height: 100vh; | ||||||
|  |   background: linear-gradient(180deg, #f8f9fa 0%, var(--background) 100%); | ||||||
|  |   color: var(--dark-gray); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .app-header { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px 20px; | ||||||
|  |   border-bottom: 1px solid #e9ecef; | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .app-nav { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: center; | ||||||
|  |   gap: 8px; | ||||||
|  |   padding: 20px; | ||||||
|  |   background: rgba(0, 0, 0, 0.02); | ||||||
|  |   border-bottom: 1px solid #e9ecef; | ||||||
|  |   margin-bottom: 40px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .nav-button { | ||||||
|  |   padding: 12px 20px; | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   background: rgba(255, 255, 255, 0.8); | ||||||
|  |   color: #6c757d; | ||||||
|  |   font-weight: 500; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   backdrop-filter: blur(10px); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .nav-button:hover { | ||||||
|  |   background: rgba(102, 126, 234, 0.1); | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1); | ||||||
|  |   color: #495057; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .nav-button.active { | ||||||
|  |   background: var(--theme-color); | ||||||
|  |   color: var(--white); | ||||||
|  |   border: 1px solid var(--theme-color); | ||||||
|  |   box-shadow: 0 4px 16px rgba(255, 69, 0, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .nav-button.active:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 6px 20px rgba(255, 69, 0, 0.5); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .app-header h1 { | ||||||
|  |   font-size: 48px; | ||||||
|  |   margin: 0; | ||||||
|  |   background: linear-gradient(90deg, #fff700 0%, #ff00ff 100%); | ||||||
|  |   -webkit-background-clip: text; | ||||||
|  |   -webkit-text-fill-color: transparent; | ||||||
|  |   background-clip: text; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .app-header p { | ||||||
|  |   color: #6c757d; | ||||||
|  |   margin-top: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-info { | ||||||
|  |   position: absolute; | ||||||
|  |   top: 20px; | ||||||
|  |   right: 20px; | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-handle { | ||||||
|  |   color: #495057; | ||||||
|  |   font-weight: bold; | ||||||
|  |   background: rgba(102, 126, 234, 0.1); | ||||||
|  |   padding: 6px 12px; | ||||||
|  |   border-radius: 20px; | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-button, | ||||||
|  | .logout-button, | ||||||
|  | .backup-button, | ||||||
|  | .token-button { | ||||||
|  |   padding: 8px 16px; | ||||||
|  |   border: none; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   font-size: 12px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   margin-left: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-button { | ||||||
|  |   background: var(--theme-color); | ||||||
|  |   color: var(--white); | ||||||
|  |   border: 1px solid var(--theme-color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .backup-button { | ||||||
|  |   background: linear-gradient(135deg, #28a745 0%, #20c997 100%); | ||||||
|  |   color: white; | ||||||
|  |   border: 1px solid #28a745; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .token-button { | ||||||
|  |   background: linear-gradient(135deg, #ffc107 0%, #fd7e14 100%); | ||||||
|  |   color: white; | ||||||
|  |   border: 1px solid #ffc107; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .logout-button { | ||||||
|  |   background: rgba(108, 117, 125, 0.1); | ||||||
|  |   color: #495057; | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(255, 69, 0, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .backup-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(40, 167, 69, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .token-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(255, 193, 7, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .logout-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1); | ||||||
|  |   background: rgba(108, 117, 125, 0.2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .loading { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   height: 100vh; | ||||||
|  |   font-size: 24px; | ||||||
|  |   color: #667eea; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .app-main { | ||||||
|  |   max-width: 1000px; | ||||||
|  |   margin: 0 auto; | ||||||
|  |   padding: 40px 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media (max-width: 1000px) { | ||||||
|  |   * { | ||||||
|  |     max-width: 100% !important; | ||||||
|  |     box-sizing: border-box !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .app .app-main { | ||||||
|  |     max-width: 100% !important; | ||||||
|  |     margin: 0 !important; | ||||||
|  |     padding: 0px !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .comment-item { | ||||||
|  |     padding: 0px !important; | ||||||
|  |     margin: 0px !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .auth-section { | ||||||
|  |     padding: 0px !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .comments-list { | ||||||
|  |     padding: 0px !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .comment-section { | ||||||
|  | 				padding: 30px 0 !important; | ||||||
|  | 				margin: 0px !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .comment-content { | ||||||
|  |     padding: 10px !important; | ||||||
|  |     word-wrap: break-word !important; | ||||||
|  |     overflow-wrap: break-word !important; | ||||||
|  |     white-space: pre-wrap !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .comment-header { | ||||||
|  |     padding: 10px !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Fix overflow on article pages */ | ||||||
|  |   article.article-content { | ||||||
|  |     overflow-x: hidden !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Ensure full width on mobile */ | ||||||
|  |   .app { | ||||||
|  |     max-width: 100vw !important; | ||||||
|  |     overflow-x: hidden !important; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Fix button overflow */ | ||||||
|  |   button { | ||||||
|  |     max-width: 100%; | ||||||
|  |     white-space: normal; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   /* Fix comment-meta URI overflow */ | ||||||
|  |   .comment-meta { | ||||||
|  |     word-break: break-all !important; | ||||||
|  |     overflow-wrap: break-word !important; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-section { | ||||||
|  |   text-align: center; | ||||||
|  |   margin-bottom: 60px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-section h2 { | ||||||
|  |   font-size: 32px; | ||||||
|  |   margin-bottom: 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-buttons { | ||||||
|  |   display: flex; | ||||||
|  |   gap: 20px; | ||||||
|  |   justify-content: center; | ||||||
|  |   flex-wrap: wrap; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-button { | ||||||
|  |   padding: 20px 40px; | ||||||
|  |   font-size: 18px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   border: none; | ||||||
|  |   border-radius: 12px; | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   color: white; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   box-shadow: 0 4px 15px rgba(0, 0, 0, 0.3); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-button:hover:not(:disabled) { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 6px 20px rgba(0, 0, 0, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-button:disabled { | ||||||
|  |   opacity: 0.5; | ||||||
|  |   cursor: not-allowed; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-button-premium { | ||||||
|  |   background: linear-gradient(135deg, #f093fb 0%, #f5576c 100%); | ||||||
|  |   position: relative; | ||||||
|  |   overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-button-premium::before { | ||||||
|  |   content: ''; | ||||||
|  |   position: absolute; | ||||||
|  |   top: -50%; | ||||||
|  |   left: -50%; | ||||||
|  |   width: 200%; | ||||||
|  |   height: 200%; | ||||||
|  |   background: linear-gradient( | ||||||
|  |     45deg, | ||||||
|  |     transparent 30%, | ||||||
|  |     rgba(255, 255, 255, 0.2) 50%, | ||||||
|  |     transparent 70% | ||||||
|  |   ); | ||||||
|  |   animation: shimmer 3s infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .collection-section h2 { | ||||||
|  |   font-size: 32px; | ||||||
|  |   text-align: center; | ||||||
|  |   margin-bottom: 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-grid { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); | ||||||
|  |   gap: 30px; | ||||||
|  |   justify-items: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .empty-message { | ||||||
|  |   text-align: center; | ||||||
|  |   color: #6c757d; | ||||||
|  |   font-size: 18px; | ||||||
|  |   margin-top: 40px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .error { | ||||||
|  |   color: #ff4757; | ||||||
|  |   text-align: center; | ||||||
|  |   margin-top: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes shimmer { | ||||||
|  |   0% { transform: translateX(-100%) rotate(45deg); } | ||||||
|  |   100% { transform: translateX(100%) rotate(45deg); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Comment System Styles */ | ||||||
|  | .comment-section { | ||||||
|  |   max-width: 800px; | ||||||
|  |   margin: 0 auto; | ||||||
|  |   /* padding: 20px; - removed to avoid double padding */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media (max-width: 768px) { | ||||||
|  |   .comment-section { | ||||||
|  |     max-width: 100%; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .auth-section { | ||||||
|  |   background: #f8f9fa; | ||||||
|  |   border: 1px solid #e9ecef; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 20px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .auth-section.search-bar-layout { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   padding: 10px; | ||||||
|  |   gap: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .auth-section.search-bar-layout .handle-input { | ||||||
|  |   flex: 1; | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 10px 15px; | ||||||
|  |   font-size: 16px; | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  |   border-radius: 6px 0 0 6px; | ||||||
|  |   background: white; | ||||||
|  |   outline: none; | ||||||
|  |   transition: border-color 0.2s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .auth-section.search-bar-layout .handle-input:focus { | ||||||
|  |   border-color: var(--theme-color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .auth-section.search-bar-layout .atproto-button { | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 10px 20px; | ||||||
|  |   border-radius: 0 6px 6px 0; | ||||||
|  |   min-width: 50px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   height: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .atproto-button { | ||||||
|  |   background: var(--theme-color); | ||||||
|  |   color: var(--white); | ||||||
|  |   border: none; | ||||||
|  |   padding: 12px 24px; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   font-size: 16px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   cursor: pointer; | ||||||
|  |   margin-bottom: 15px; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .atproto-button:hover { | ||||||
|  |   filter: brightness(1.1); | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(255, 69, 0, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .username-input-section { | ||||||
|  |   margin: 15px 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .handle-input { | ||||||
|  |   width: 300px; | ||||||
|  |   max-width: 100%; | ||||||
|  |   padding: 10px; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   font-size: 14px; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Override for search bar layout */ | ||||||
|  | .search-bar-layout .handle-input { | ||||||
|  |   width: auto; | ||||||
|  |   text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Mobile responsive for search bar */ | ||||||
|  | @media (max-width: 480px) { | ||||||
|  |   .auth-section.search-bar-layout { | ||||||
|  |     flex-direction: column; | ||||||
|  |     gap: 8px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .auth-section.search-bar-layout .handle-input { | ||||||
|  |     width: 100%; | ||||||
|  |     border-radius: 6px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .auth-section.search-bar-layout .atproto-button { | ||||||
|  |     width: 100%; | ||||||
|  |     border-radius: 6px; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .auth-hint { | ||||||
|  |   color: #6c757d; | ||||||
|  |   font-size: 14px; | ||||||
|  |   margin: 10px 0 0 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-section { | ||||||
|  |   background: #e8f5e8; | ||||||
|  |   border: 1px solid #4caf50; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 20px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-section .user-info { | ||||||
|  |   position: static; | ||||||
|  |   display: block; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-profile { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 15px; | ||||||
|  |   margin-bottom: 15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-avatar { | ||||||
|  |   width: 48px; | ||||||
|  |   height: 48px; | ||||||
|  |   border-radius: 50%; | ||||||
|  |   object-fit: cover; | ||||||
|  |   border: 2px solid #4caf50; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-details h3 { | ||||||
|  |   margin: 0 0 5px 0; | ||||||
|  |   color: #333; | ||||||
|  |   font-size: 18px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-section .user-info h3 { | ||||||
|  |   margin: 0 0 10px 0; | ||||||
|  |   color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-section .user-handle { | ||||||
|  |   background: rgba(76, 175, 80, 0.1); | ||||||
|  |   color: #2e7d32; | ||||||
|  |   border: 1px solid #4caf50; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-section .user-did { | ||||||
|  |   font-family: monospace; | ||||||
|  |   font-size: 0.8em; | ||||||
|  |   color: #666; | ||||||
|  |   background: #f1f3f4; | ||||||
|  |   padding: 4px 8px; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   margin-top: 5px; | ||||||
|  |   word-break: break-all; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-form { | ||||||
|  |   background: #fff; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 20px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-form h3 { | ||||||
|  |   margin: 0 0 15px 0; | ||||||
|  |   color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-form textarea { | ||||||
|  |   width: 100%; | ||||||
|  |   padding: 12px; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   font-family: inherit; | ||||||
|  |   font-size: 14px; | ||||||
|  |   resize: vertical; | ||||||
|  |   box-sizing: border-box; | ||||||
|  |   min-height: 100px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-form textarea:focus { | ||||||
|  |   border-color: #1185fe; | ||||||
|  |   outline: none; | ||||||
|  |   box-shadow: 0 0 0 2px rgba(17, 133, 254, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-actions { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   align-items: center; | ||||||
|  |   margin-top: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .char-count { | ||||||
|  |   color: #666; | ||||||
|  |   font-size: 0.9em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-button { | ||||||
|  |   background: var(--theme-color); | ||||||
|  |   color: var(--white); | ||||||
|  |   border: none; | ||||||
|  |   padding: 10px 20px; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 14px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-button:hover:not(:disabled) { | ||||||
|  |   filter: brightness(1.1); | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(255, 69, 0, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .post-button:disabled { | ||||||
|  |   background: #6c757d; | ||||||
|  |   cursor: not-allowed; | ||||||
|  |   transform: none; | ||||||
|  |   box-shadow: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comments-list { | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 0px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comments-header { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   align-items: center; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comments-header h3 { | ||||||
|  |   margin: 0; | ||||||
|  |   color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comments-controls { | ||||||
|  |   display: flex; | ||||||
|  |   gap: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comments-toggle-button { | ||||||
|  |   background: var(--theme-color); | ||||||
|  |   color: var(--white); | ||||||
|  |   border: none; | ||||||
|  |   padding: 8px 16px; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 14px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comments-toggle-button:hover { | ||||||
|  |   filter: brightness(1.1); | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(255, 69, 0, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-item { | ||||||
|  |   border: 1px solid #e9ecef; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   padding: 15px; | ||||||
|  |   margin-bottom: 15px; | ||||||
|  |   background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-item:last-child { | ||||||
|  |   margin-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-header { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 10px; | ||||||
|  |   margin-bottom: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-avatar { | ||||||
|  |   width: 32px; | ||||||
|  |   height: 32px; | ||||||
|  |   border-radius: 50%; | ||||||
|  |   object-fit: cover; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-author-info { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   gap: 2px; | ||||||
|  |   flex: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-author { | ||||||
|  |   font-weight: bold; | ||||||
|  |   color: #333; | ||||||
|  |   font-size: 0.95em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-handle { | ||||||
|  |   color: #666; | ||||||
|  |   font-size: 0.8em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-date { | ||||||
|  |   color: #666; | ||||||
|  |   font-size: 0.9em; | ||||||
|  |   margin-left: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-button { | ||||||
|  |   background: #dc3545; | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 12px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   padding: 4px 8px; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   margin-left: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-button:hover { | ||||||
|  |   background: #c82333; | ||||||
|  |   transform: scale(1.05); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-content { | ||||||
|  |   line-height: 1.5; | ||||||
|  |   color: #333; | ||||||
|  |   margin-bottom: 10px; | ||||||
|  |   white-space: pre-wrap; | ||||||
|  |   word-wrap: break-word; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-meta { | ||||||
|  |   padding: 8px; | ||||||
|  |   background: #f1f3f4; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   font-size: 0.8em; | ||||||
|  |   color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .comment-meta small { | ||||||
|  |   font-family: monospace; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .no-comments { | ||||||
|  |   text-align: center; | ||||||
|  |   color: #666; | ||||||
|  |   font-style: italic; | ||||||
|  |   padding: 40px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .error { | ||||||
|  |   background: #f8d7da; | ||||||
|  |   color: #721c24; | ||||||
|  |   border: 1px solid #f5c6cb; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   padding: 10px; | ||||||
|  |   margin-top: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Admin Section Styles */ | ||||||
|  | .admin-section { | ||||||
|  |   background: #e3f2fd; | ||||||
|  |   border: 1px solid #2196f3; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 20px; | ||||||
|  |   margin-top: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .admin-section h3 { | ||||||
|  |   margin: 0 0 15px 0; | ||||||
|  |   color: #1976d2; | ||||||
|  |   font-size: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-form { | ||||||
|  |   background: #fff; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-form textarea { | ||||||
|  |   width: 100%; | ||||||
|  |   padding: 12px; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   font-family: inherit; | ||||||
|  |   font-size: 14px; | ||||||
|  |   resize: vertical; | ||||||
|  |   box-sizing: border-box; | ||||||
|  |   min-height: 80px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-form textarea:focus { | ||||||
|  |   border-color: #2196f3; | ||||||
|  |   outline: none; | ||||||
|  |   box-shadow: 0 0 0 2px rgba(33, 150, 243, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .admin-hint { | ||||||
|  |   color: #666; | ||||||
|  |   font-size: 0.9em; | ||||||
|  |   font-style: italic; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* User List Records Styles */ | ||||||
|  | .user-list-records { | ||||||
|  |   margin-top: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-records h4 { | ||||||
|  |   margin: 0 0 15px 0; | ||||||
|  |   color: #1976d2; | ||||||
|  |   font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .no-user-lists { | ||||||
|  |   text-align: center; | ||||||
|  |   color: #666; | ||||||
|  |   font-style: italic; | ||||||
|  |   padding: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-item { | ||||||
|  |   border: 1px solid #e3f2fd; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   padding: 12px; | ||||||
|  |   margin-bottom: 10px; | ||||||
|  |   background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-item:last-child { | ||||||
|  |   margin-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-header { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   align-items: center; | ||||||
|  |   margin-bottom: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-actions { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-date { | ||||||
|  |   color: #666; | ||||||
|  |   font-size: 0.9em; | ||||||
|  |   font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-content { | ||||||
|  |   margin-top: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-handles { | ||||||
|  |   display: flex; | ||||||
|  |   flex-wrap: wrap; | ||||||
|  |   gap: 8px; | ||||||
|  |   margin-bottom: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-handle-tag { | ||||||
|  |   background: #e3f2fd; | ||||||
|  |   color: #1976d2; | ||||||
|  |   padding: 4px 8px; | ||||||
|  |   border-radius: 12px; | ||||||
|  |   font-size: 0.85em; | ||||||
|  |   font-weight: 500; | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .pds-info { | ||||||
|  |   color: #666; | ||||||
|  |   font-size: 0.75em; | ||||||
|  |   font-weight: normal; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-meta { | ||||||
|  |   font-size: 0.8em; | ||||||
|  |   color: #666; | ||||||
|  |   background: #f8f9fa; | ||||||
|  |   padding: 6px 8px; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   line-height: 1.4; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .user-list-meta small { | ||||||
|  |   font-family: monospace; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* JSON Display Styles */ | ||||||
|  | .json-button { | ||||||
|  |   background: var(--theme-color); | ||||||
|  |   color: var(--white); | ||||||
|  |   border: none; | ||||||
|  |   padding: 4px 8px; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 12px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-button:hover { | ||||||
|  |   filter: brightness(1.1); | ||||||
|  |   transform: scale(1.05); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-display { | ||||||
|  |   margin-top: 12px; | ||||||
|  |   border: 1px solid #ddd; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-display h5 { | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 8px 12px; | ||||||
|  |   background: #f1f3f4; | ||||||
|  |   border-bottom: 1px solid #ddd; | ||||||
|  |   font-size: 0.9em; | ||||||
|  |   color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-content { | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 12px; | ||||||
|  |   background: #f8f9fa; | ||||||
|  |   font-family: 'Monaco', 'Menlo', 'Ubuntu Mono', monospace; | ||||||
|  |   font-size: 0.8em; | ||||||
|  |   line-height: 1.4; | ||||||
|  |   overflow-x: auto; | ||||||
|  |   white-space: pre-wrap; | ||||||
|  |   word-break: break-word; | ||||||
|  |   color: #333; | ||||||
|  |   max-height: 400px; | ||||||
|  |   overflow-y: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Tab Navigation */ | ||||||
|  | .tab-navigation { | ||||||
|  |   display: flex; | ||||||
|  |   border-bottom: 2px solid #e1e5e9; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .tab-button { | ||||||
|  |   background: none; | ||||||
|  |   border: none; | ||||||
|  |   padding: 12px 20px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 14px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   color: #656d76; | ||||||
|  |   border-bottom: 2px solid transparent; | ||||||
|  |   transition: all 0.2s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .tab-button:hover { | ||||||
|  |   color: var(--theme-color); | ||||||
|  |   background: #f6f8fa; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .tab-button.active { | ||||||
|  |   color: var(--theme-color); | ||||||
|  |   border-bottom-color: var(--theme-color); | ||||||
|  |   background: #f6f8fa; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .chat-actions { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .chat-type-button { | ||||||
|  |   background: var(--theme-color); | ||||||
|  |   color: var(--white); | ||||||
|  |   border: none; | ||||||
|  |   padding: 4px 8px; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   cursor: default; | ||||||
|  |   font-size: 12px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   margin-left: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .chat-type-text { | ||||||
|  |   font-size: 16px; | ||||||
|  |   margin-left: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .chat-date { | ||||||
|  |   color: #656d76; | ||||||
|  |   font-size: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .chat-content { | ||||||
|  |   background: #f6f8fa; | ||||||
|  |   padding: 12px; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   border-left: 4px solid #d1d9e0; | ||||||
|  |   margin-bottom: 8px; | ||||||
|  |   white-space: pre-wrap; | ||||||
|  |   line-height: 1.5; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .chat-meta { | ||||||
|  |   font-size: 11px; | ||||||
|  |   color: #656d76; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .no-chat { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px 20px; | ||||||
|  |   color: #656d76; | ||||||
|  |   font-style: italic; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .chat-message.comment-style { | ||||||
|  |   border-left: 4px solid var(--theme-color); | ||||||
|  | } | ||||||
							
								
								
									
										1631
									
								
								oauth/src/App.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1631
									
								
								oauth/src/App.tsx
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										21
									
								
								oauth/src/components/AIChat-access.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								oauth/src/components/AIChat-access.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | // Cloudflare Access対応版の例 | ||||||
|  | const response = await fetch(`${aiConfig.host}/api/generate`, { | ||||||
|  |   method: 'POST', | ||||||
|  |   headers: { | ||||||
|  |     'Content-Type': 'application/json', | ||||||
|  |     // Cloudflare Access Service Token | ||||||
|  |     'CF-Access-Client-Id': import.meta.env.VITE_CF_ACCESS_CLIENT_ID, | ||||||
|  |     'CF-Access-Client-Secret': import.meta.env.VITE_CF_ACCESS_CLIENT_SECRET, | ||||||
|  |   }, | ||||||
|  |   body: JSON.stringify({ | ||||||
|  |     model: aiConfig.model, | ||||||
|  |     prompt: prompt, | ||||||
|  |     stream: false, | ||||||
|  |     options: { | ||||||
|  |       temperature: 0.9, | ||||||
|  |       top_p: 0.9, | ||||||
|  |       num_predict: 200, | ||||||
|  |       repeat_penalty: 1.1, | ||||||
|  |     } | ||||||
|  |   }), | ||||||
|  | }); | ||||||
							
								
								
									
										271
									
								
								oauth/src/components/AIChat.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										271
									
								
								oauth/src/components/AIChat.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,271 @@ | |||||||
|  | import React, { useState, useEffect } from 'react'; | ||||||
|  | import { User } from '../services/auth'; | ||||||
|  | import { atprotoOAuthService } from '../services/atproto-oauth'; | ||||||
|  | import { appConfig, getCollectionNames } from '../config/app'; | ||||||
|  |  | ||||||
|  | interface AIChatProps { | ||||||
|  |   user: User | null; | ||||||
|  |   isEnabled: boolean; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const AIChat: React.FC<AIChatProps> = ({ user, isEnabled }) => { | ||||||
|  |   const [chatHistory, setChatHistory] = useState<any[]>([]); | ||||||
|  |   const [isLoading, setIsLoading] = useState(false); | ||||||
|  |   const [isProcessing, setIsProcessing] = useState(false); | ||||||
|  |   const [aiProfile, setAiProfile] = useState<any>(null); | ||||||
|  |    | ||||||
|  |   // Get AI settings from appConfig (unified configuration) | ||||||
|  |   const aiConfig = { | ||||||
|  |     enabled: appConfig.aiEnabled, | ||||||
|  |     askAi: appConfig.aiAskAi, | ||||||
|  |     provider: appConfig.aiProvider, | ||||||
|  |     model: appConfig.aiModel, | ||||||
|  |     host: appConfig.aiHost, | ||||||
|  |     systemPrompt: appConfig.aiSystemPrompt, | ||||||
|  |     aiDid: appConfig.aiDid, | ||||||
|  |     bskyPublicApi: appConfig.bskyPublicApi, | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // Fetch AI profile on load | ||||||
|  |   useEffect(() => { | ||||||
|  |     const fetchAIProfile = async () => { | ||||||
|  |       if (!aiConfig.aiDid) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       try { | ||||||
|  |         // Try with agent first | ||||||
|  |         const agent = atprotoOAuthService.getAgent(); | ||||||
|  |         if (agent) { | ||||||
|  |           const profile = await agent.getProfile({ actor: aiConfig.aiDid }); | ||||||
|  |           const profileData = { | ||||||
|  |             did: aiConfig.aiDid, | ||||||
|  |             handle: profile.data.handle, | ||||||
|  |             displayName: profile.data.displayName, | ||||||
|  |             avatar: profile.data.avatar, | ||||||
|  |             description: profile.data.description | ||||||
|  |           }; | ||||||
|  |           setAiProfile(profileData); | ||||||
|  |            | ||||||
|  |           // Dispatch event to update Ask AI button | ||||||
|  |           window.dispatchEvent(new CustomEvent('aiProfileLoaded', { detail: profileData })); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // Fallback to public API | ||||||
|  |         const response = await fetch(`${aiConfig.bskyPublicApi}/xrpc/app.bsky.actor.getProfile?actor=${encodeURIComponent(aiConfig.aiDid)}`); | ||||||
|  |         if (response.ok) { | ||||||
|  |           const profileData = await response.json(); | ||||||
|  |           const profile = { | ||||||
|  |             did: aiConfig.aiDid, | ||||||
|  |             handle: profileData.handle, | ||||||
|  |             displayName: profileData.displayName, | ||||||
|  |             avatar: profileData.avatar, | ||||||
|  |             description: profileData.description | ||||||
|  |           }; | ||||||
|  |           setAiProfile(profile); | ||||||
|  |            | ||||||
|  |           // Dispatch event to update Ask AI button | ||||||
|  |           window.dispatchEvent(new CustomEvent('aiProfileLoaded', { detail: profile })); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |       } catch (error) { | ||||||
|  |         setAiProfile(null); | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     fetchAIProfile(); | ||||||
|  |   }, [aiConfig.aiDid]); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (!isEnabled || !aiConfig.askAi) return; | ||||||
|  |  | ||||||
|  |     // Listen for AI question posts from base.html | ||||||
|  |     const handleAIQuestion = async (event: any) => { | ||||||
|  |       if (!user || !event.detail || !event.detail.question || isProcessing || !aiProfile) return; | ||||||
|  |        | ||||||
|  |       setIsProcessing(true); | ||||||
|  |       try { | ||||||
|  |         await postQuestionAndGenerateResponse(event.detail.question); | ||||||
|  |       } finally { | ||||||
|  |         setIsProcessing(false); | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // Add listener with a small delay to ensure it's ready | ||||||
|  |     setTimeout(() => { | ||||||
|  |       window.addEventListener('postAIQuestion', handleAIQuestion); | ||||||
|  |        | ||||||
|  |       // Notify that AI is ready | ||||||
|  |       window.dispatchEvent(new CustomEvent('aiChatReady')); | ||||||
|  |     }, 100); | ||||||
|  |  | ||||||
|  |     return () => { | ||||||
|  |       window.removeEventListener('postAIQuestion', handleAIQuestion); | ||||||
|  |     }; | ||||||
|  |   }, [user, isEnabled, isProcessing, aiProfile]); | ||||||
|  |  | ||||||
|  |   const postQuestionAndGenerateResponse = async (question: string) => { | ||||||
|  |     if (!user || !aiConfig.askAi || !aiProfile) return; | ||||||
|  |  | ||||||
|  |     setIsLoading(true); | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       const agent = atprotoOAuthService.getAgent(); | ||||||
|  |       if (!agent) throw new Error('No agent available'); | ||||||
|  |  | ||||||
|  |       // Get collection names | ||||||
|  |       const collections = getCollectionNames(appConfig.collections.base); | ||||||
|  |  | ||||||
|  |       // 1. Post question to ATProto | ||||||
|  |       const now = new Date(); | ||||||
|  |       const rkey = now.toISOString().replace(/[:.]/g, '-'); | ||||||
|  |        | ||||||
|  |       // Extract post metadata from current page | ||||||
|  |       const currentUrl = window.location.href; | ||||||
|  |       const postSlug = currentUrl.match(/\/posts\/([^/]+)/)?.[1] || ''; | ||||||
|  |       const postTitle = document.title.replace(' - syui.ai', '') || ''; | ||||||
|  |        | ||||||
|  |       const questionRecord = { | ||||||
|  |         $type: collections.chat, | ||||||
|  |         post: { | ||||||
|  |           url: currentUrl, | ||||||
|  |           slug: postSlug, | ||||||
|  |           title: postTitle, | ||||||
|  |           date: new Date().toISOString(), | ||||||
|  |           tags: [], | ||||||
|  |           language: "ja" | ||||||
|  |         }, | ||||||
|  |         type: "question", | ||||||
|  |         text: question, | ||||||
|  |         author: { | ||||||
|  |           did: user.did, | ||||||
|  |           handle: user.handle, | ||||||
|  |           avatar: user.avatar, | ||||||
|  |           displayName: user.displayName || user.handle, | ||||||
|  |         }, | ||||||
|  |         createdAt: now.toISOString(), | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       await agent.api.com.atproto.repo.putRecord({ | ||||||
|  |         repo: user.did, | ||||||
|  |         collection: collections.chat, | ||||||
|  |         rkey: rkey, | ||||||
|  |         record: questionRecord, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       // 2. Get chat history | ||||||
|  |       const chatRecords = await agent.api.com.atproto.repo.listRecords({ | ||||||
|  |         repo: user.did, | ||||||
|  |         collection: collections.chat, | ||||||
|  |         limit: 10, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       let chatHistoryText = ''; | ||||||
|  |       if (chatRecords.data.records) { | ||||||
|  |         chatHistoryText = chatRecords.data.records | ||||||
|  |           .map((r: any) => { | ||||||
|  |             if (r.value.type === 'question') { | ||||||
|  |               return `User: ${r.value.text}`; | ||||||
|  |             } else if (r.value.type === 'answer') { | ||||||
|  |               return `AI: ${r.value.text}`; | ||||||
|  |             } | ||||||
|  |             return ''; | ||||||
|  |           }) | ||||||
|  |           .filter(Boolean) | ||||||
|  |           .join('\n'); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // 3. Generate AI response based on provider | ||||||
|  |       let aiAnswer = ''; | ||||||
|  |        | ||||||
|  |       // 3. Generate AI response using Ollama via proxy | ||||||
|  |       if (aiConfig.provider === 'ollama') { | ||||||
|  |         const prompt = `${aiConfig.systemPrompt} | ||||||
|  |  | ||||||
|  | Question: ${question} | ||||||
|  |  | ||||||
|  | Answer:`; | ||||||
|  |  | ||||||
|  |         const response = await fetch(`${aiConfig.host}/api/generate`, { | ||||||
|  |           method: 'POST', | ||||||
|  |           headers: { | ||||||
|  |             'Content-Type': 'application/json', | ||||||
|  |             'Origin': 'https://syui.ai', | ||||||
|  |           }, | ||||||
|  |           body: JSON.stringify({ | ||||||
|  |             model: aiConfig.model, | ||||||
|  |             prompt: prompt, | ||||||
|  |             stream: false, | ||||||
|  |             options: { | ||||||
|  |               temperature: 0.9, | ||||||
|  |               top_p: 0.9, | ||||||
|  |               num_predict: 200, // Longer responses for better answers | ||||||
|  |               repeat_penalty: 1.1, | ||||||
|  |             } | ||||||
|  |           }), | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         if (!response.ok) { | ||||||
|  |           throw new Error('AI API request failed'); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const data = await response.json(); | ||||||
|  |         aiAnswer = data.response; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // 4. Immediately dispatch event to update UI | ||||||
|  |       window.dispatchEvent(new CustomEvent('aiResponseReceived', { | ||||||
|  |         detail: {  | ||||||
|  |           answer: aiAnswer, | ||||||
|  |           aiProfile: aiProfile, | ||||||
|  |           timestamp: now.toISOString() | ||||||
|  |         } | ||||||
|  |       })); | ||||||
|  |  | ||||||
|  |       // 5. Save AI response in background | ||||||
|  |       const answerRkey = now.toISOString().replace(/[:.]/g, '-') + '-answer'; | ||||||
|  |        | ||||||
|  |       const answerRecord = { | ||||||
|  |         $type: collections.chat, | ||||||
|  |         post: { | ||||||
|  |           url: currentUrl, | ||||||
|  |           slug: postSlug, | ||||||
|  |           title: postTitle, | ||||||
|  |           date: new Date().toISOString(), | ||||||
|  |           tags: [], | ||||||
|  |           language: "ja" | ||||||
|  |         }, | ||||||
|  |         type: "answer", | ||||||
|  |         text: aiAnswer, | ||||||
|  |         author: { | ||||||
|  |           did: aiProfile.did, | ||||||
|  |           handle: aiProfile.handle, | ||||||
|  |           displayName: aiProfile.displayName, | ||||||
|  |           avatar: aiProfile.avatar, | ||||||
|  |         }, | ||||||
|  |         createdAt: now.toISOString(), | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       // Save to ATProto asynchronously (don't wait for it) | ||||||
|  |       agent.api.com.atproto.repo.putRecord({ | ||||||
|  |         repo: user.did, | ||||||
|  |         collection: collections.chat, | ||||||
|  |         rkey: answerRkey, | ||||||
|  |         record: answerRecord, | ||||||
|  |       }).catch(err => { | ||||||
|  |         // Silent fail for AI response saving | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |     } catch (error) { | ||||||
|  |       window.dispatchEvent(new CustomEvent('aiResponseError', { | ||||||
|  |         detail: { error: 'AI応答の生成に失敗しました' } | ||||||
|  |       })); | ||||||
|  |     } finally { | ||||||
|  |       setIsLoading(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // This component doesn't render anything - it just handles the logic | ||||||
|  |   return null; | ||||||
|  | }; | ||||||
							
								
								
									
										79
									
								
								oauth/src/components/AIProfile.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								oauth/src/components/AIProfile.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | |||||||
|  | import React, { useState, useEffect } from 'react'; | ||||||
|  | import { AtprotoAgent } from '@atproto/api'; | ||||||
|  |  | ||||||
|  | interface AIProfile { | ||||||
|  |   did: string; | ||||||
|  |   handle: string; | ||||||
|  |   displayName?: string; | ||||||
|  |   avatar?: string; | ||||||
|  |   description?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface AIProfileProps { | ||||||
|  |   aiDid: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const AIProfile: React.FC<AIProfileProps> = ({ aiDid }) => { | ||||||
|  |   const [profile, setProfile] = useState<AIProfile | null>(null); | ||||||
|  |   const [loading, setLoading] = useState(true); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     const fetchAIProfile = async () => { | ||||||
|  |       try { | ||||||
|  |         // Use public API to get profile information | ||||||
|  |         const agent = new AtprotoAgent({ service: 'https://bsky.social' }); | ||||||
|  |         const response = await agent.getProfile({ actor: aiDid }); | ||||||
|  |          | ||||||
|  |         setProfile({ | ||||||
|  |           did: response.data.did, | ||||||
|  |           handle: response.data.handle, | ||||||
|  |           displayName: response.data.displayName, | ||||||
|  |           avatar: response.data.avatar, | ||||||
|  |           description: response.data.description, | ||||||
|  |         }); | ||||||
|  |       } catch (error) { | ||||||
|  |         // Failed to fetch AI profile | ||||||
|  |         // Fallback to basic info | ||||||
|  |         setProfile({ | ||||||
|  |           did: aiDid, | ||||||
|  |           handle: 'ai-assistant', | ||||||
|  |           displayName: 'AI Assistant', | ||||||
|  |           description: 'AI assistant for this blog', | ||||||
|  |         }); | ||||||
|  |       } finally { | ||||||
|  |         setLoading(false); | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if (aiDid) { | ||||||
|  |       fetchAIProfile(); | ||||||
|  |     } | ||||||
|  |   }, [aiDid]); | ||||||
|  |  | ||||||
|  |   if (loading) { | ||||||
|  |     return <div className="ai-profile-loading">Loading AI profile...</div>; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (!profile) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div className="ai-profile"> | ||||||
|  |       <div className="ai-avatar"> | ||||||
|  |         {profile.avatar ? ( | ||||||
|  |           <img src={profile.avatar} alt={profile.displayName || profile.handle} /> | ||||||
|  |         ) : ( | ||||||
|  |           <div className="ai-avatar-placeholder">🤖</div> | ||||||
|  |         )} | ||||||
|  |       </div> | ||||||
|  |       <div className="ai-info"> | ||||||
|  |         <div className="ai-name">{profile.displayName || profile.handle}</div> | ||||||
|  |         <div className="ai-handle">@{profile.handle}</div> | ||||||
|  |         {profile.description && ( | ||||||
|  |           <div className="ai-description">{profile.description}</div> | ||||||
|  |         )} | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										120
									
								
								oauth/src/components/Card.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								oauth/src/components/Card.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,120 @@ | |||||||
|  | import React from 'react'; | ||||||
|  | import { motion } from 'framer-motion'; | ||||||
|  | import { Card as CardType, CardRarity } from '../types/card'; | ||||||
|  | import '../styles/Card.css'; | ||||||
|  |  | ||||||
|  | interface CardProps { | ||||||
|  |   card: CardType; | ||||||
|  |   isRevealing?: boolean; | ||||||
|  |   detailed?: boolean; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const CARD_INFO: Record<number, { name: string; color: string }> = { | ||||||
|  |   0: { name: "アイ", color: "#fff700" }, | ||||||
|  |   1: { name: "夢幻", color: "#b19cd9" }, | ||||||
|  |   2: { name: "光彩", color: "#ffd700" }, | ||||||
|  |   3: { name: "中性子", color: "#cacfd2" }, | ||||||
|  |   4: { name: "太陽", color: "#ff6b35" }, | ||||||
|  |   5: { name: "夜空", color: "#1a1a2e" }, | ||||||
|  |   6: { name: "雪", color: "#e3f2fd" }, | ||||||
|  |   7: { name: "雷", color: "#ffd93d" }, | ||||||
|  |   8: { name: "超究", color: "#6c5ce7" }, | ||||||
|  |   9: { name: "剣", color: "#a8e6cf" }, | ||||||
|  |   10: { name: "破壊", color: "#ff4757" }, | ||||||
|  |   11: { name: "地球", color: "#4834d4" }, | ||||||
|  |   12: { name: "天の川", color: "#9c88ff" }, | ||||||
|  |   13: { name: "創造", color: "#00d2d3" }, | ||||||
|  |   14: { name: "超新星", color: "#ff9ff3" }, | ||||||
|  |   15: { name: "世界", color: "#54a0ff" }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const Card: React.FC<CardProps> = ({ card, isRevealing = false, detailed = false }) => { | ||||||
|  |   const cardInfo = CARD_INFO[card.id] || { name: "Unknown", color: "#666" }; | ||||||
|  |   const imageUrl = `https://git.syui.ai/ai/card/raw/branch/main/img/${card.id}.webp`; | ||||||
|  |    | ||||||
|  |   const getRarityClass = () => { | ||||||
|  |     switch (card.status) { | ||||||
|  |       case CardRarity.UNIQUE: | ||||||
|  |         return 'card-unique'; | ||||||
|  |       case CardRarity.KIRA: | ||||||
|  |         return 'card-kira'; | ||||||
|  |       case CardRarity.SUPER_RARE: | ||||||
|  |         return 'card-super-rare'; | ||||||
|  |       case CardRarity.RARE: | ||||||
|  |         return 'card-rare'; | ||||||
|  |       default: | ||||||
|  |         return 'card-normal'; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   if (!detailed) { | ||||||
|  |     // Simple view - only image and frame | ||||||
|  |     return ( | ||||||
|  |       <motion.div | ||||||
|  |         className={`card card-simple ${getRarityClass()}`} | ||||||
|  |         initial={isRevealing ? { rotateY: 180 } : {}} | ||||||
|  |         animate={isRevealing ? { rotateY: 0 } : {}} | ||||||
|  |         transition={{ duration: 0.8, type: "spring" }} | ||||||
|  |       > | ||||||
|  |         <div className="card-frame"> | ||||||
|  |           <img  | ||||||
|  |             src={imageUrl}  | ||||||
|  |             alt={cardInfo.name} | ||||||
|  |             className="card-image-simple" | ||||||
|  |             onError={(e) => { | ||||||
|  |               (e.target as HTMLImageElement).style.display = 'none'; | ||||||
|  |             }} | ||||||
|  |           /> | ||||||
|  |         </div> | ||||||
|  |       </motion.div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Detailed view - all information | ||||||
|  |   return ( | ||||||
|  |     <motion.div | ||||||
|  |       className={`card ${getRarityClass()}`} | ||||||
|  |       initial={isRevealing ? { rotateY: 180 } : {}} | ||||||
|  |       animate={isRevealing ? { rotateY: 0 } : {}} | ||||||
|  |       transition={{ duration: 0.8, type: "spring" }} | ||||||
|  |       style={{ | ||||||
|  |         '--card-color': cardInfo.color, | ||||||
|  |       } as React.CSSProperties} | ||||||
|  |     > | ||||||
|  |       <div className="card-inner"> | ||||||
|  |         <div className="card-header"> | ||||||
|  |           <span className="card-id">#{card.id}</span> | ||||||
|  |           <span className="card-cp">CP: {card.cp}</span> | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         <div className="card-image-container"> | ||||||
|  |           <img  | ||||||
|  |             src={imageUrl}  | ||||||
|  |             alt={cardInfo.name} | ||||||
|  |             className="card-image" | ||||||
|  |             onError={(e) => { | ||||||
|  |               (e.target as HTMLImageElement).style.display = 'none'; | ||||||
|  |             }} | ||||||
|  |           /> | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         <div className="card-content"> | ||||||
|  |           <h3 className="card-name">{cardInfo.name}</h3> | ||||||
|  |           {card.is_unique && ( | ||||||
|  |             <div className="unique-badge">UNIQUE</div> | ||||||
|  |           )} | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         {card.skill && ( | ||||||
|  |           <div className="card-skill"> | ||||||
|  |             <p>{card.skill}</p> | ||||||
|  |           </div> | ||||||
|  |         )} | ||||||
|  |          | ||||||
|  |         <div className="card-footer"> | ||||||
|  |           <span className="card-rarity">{card.status.toUpperCase()}</span> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </motion.div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										171
									
								
								oauth/src/components/CardBox.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										171
									
								
								oauth/src/components/CardBox.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,171 @@ | |||||||
|  | import React, { useState, useEffect } from 'react'; | ||||||
|  | import { atprotoOAuthService } from '../services/atproto-oauth'; | ||||||
|  | import { Card } from './Card'; | ||||||
|  | import '../styles/CardBox.css'; | ||||||
|  |  | ||||||
|  | interface CardBoxProps { | ||||||
|  |   userDid: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const CardBox: React.FC<CardBoxProps> = ({ userDid }) => { | ||||||
|  |   const [boxData, setBoxData] = useState<any>(null); | ||||||
|  |   const [loading, setLoading] = useState(true); | ||||||
|  |   const [error, setError] = useState<string | null>(null); | ||||||
|  |   const [showJson, setShowJson] = useState(false); | ||||||
|  |   const [isDeleting, setIsDeleting] = useState(false); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     loadBoxData(); | ||||||
|  |   }, [userDid]); | ||||||
|  |  | ||||||
|  |   const loadBoxData = async () => { | ||||||
|  |     setLoading(true); | ||||||
|  |     setError(null); | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       const data = await atprotoOAuthService.getCardsFromBox(); | ||||||
|  |       setBoxData(data); | ||||||
|  |     } catch (err) { | ||||||
|  |       // Failed to load card box | ||||||
|  |       setError(err instanceof Error ? err.message : 'カードボックスの読み込みに失敗しました'); | ||||||
|  |     } finally { | ||||||
|  |       setLoading(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleSaveToBox = async () => { | ||||||
|  |     // 現在のカードデータを取得してボックスに保存 | ||||||
|  |     // この部分は親コンポーネントから渡すか、APIから取得する必要があります | ||||||
|  |     alert('カードボックスへの保存機能は親コンポーネントから実行してください'); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleDeleteBox = async () => { | ||||||
|  |     if (!window.confirm('カードボックスを削除してもよろしいですか?\nこの操作は取り消せません。')) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     setIsDeleting(true); | ||||||
|  |     setError(null); | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       await atprotoOAuthService.deleteCardBox(); | ||||||
|  |       setBoxData({ records: [] }); | ||||||
|  |       alert('カードボックスを削除しました'); | ||||||
|  |     } catch (err) { | ||||||
|  |       // Failed to delete card box | ||||||
|  |       setError(err instanceof Error ? err.message : 'カードボックスの削除に失敗しました'); | ||||||
|  |     } finally { | ||||||
|  |       setIsDeleting(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   if (loading) { | ||||||
|  |     return ( | ||||||
|  |       <div className="card-box-container"> | ||||||
|  |         <div className="loading">カードボックスを読み込み中...</div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (error) { | ||||||
|  |     return ( | ||||||
|  |       <div className="card-box-container"> | ||||||
|  |         <div className="error">エラー: {error}</div> | ||||||
|  |         <button onClick={loadBoxData} className="retry-button"> | ||||||
|  |           再試行 | ||||||
|  |         </button> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const records = boxData?.records || []; | ||||||
|  |   const selfRecord = records.find((record: any) => record.uri.includes('/self')); | ||||||
|  |   const cards = selfRecord?.value?.cards || []; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div className="card-box-container"> | ||||||
|  |       <div className="card-box-header"> | ||||||
|  |         <h3>📦 atproto カードボックス</h3> | ||||||
|  |         <div className="box-actions"> | ||||||
|  |           <button  | ||||||
|  |             onClick={() => setShowJson(!showJson)}  | ||||||
|  |             className="json-button" | ||||||
|  |           > | ||||||
|  |             {showJson ? 'JSON非表示' : 'JSON表示'} | ||||||
|  |           </button> | ||||||
|  |           <button onClick={loadBoxData} className="refresh-button"> | ||||||
|  |             🔄 更新 | ||||||
|  |           </button> | ||||||
|  |           {cards.length > 0 && ( | ||||||
|  |             <button  | ||||||
|  |               onClick={handleDeleteBox}  | ||||||
|  |               className="delete-button" | ||||||
|  |               disabled={isDeleting} | ||||||
|  |             > | ||||||
|  |               {isDeleting ? '削除中...' : '🗑️ 削除'} | ||||||
|  |             </button> | ||||||
|  |           )} | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       <div className="uri-display"> | ||||||
|  |         <p> | ||||||
|  |           <strong>📍 URI:</strong>  | ||||||
|  |           <code>at://did:plc:uqzpqmrjnptsxezjx4xuh2mn/ai.card.box/self</code> | ||||||
|  |         </p> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       {showJson && ( | ||||||
|  |         <div className="json-display"> | ||||||
|  |           <h4>Raw JSON データ:</h4> | ||||||
|  |           <pre className="json-content"> | ||||||
|  |             {JSON.stringify(boxData, null, 2)} | ||||||
|  |           </pre> | ||||||
|  |         </div> | ||||||
|  |       )} | ||||||
|  |  | ||||||
|  |       <div className="box-stats"> | ||||||
|  |         <p> | ||||||
|  |           <strong>総カード数:</strong> {cards.length}枚 | ||||||
|  |           {selfRecord?.value?.updated_at && ( | ||||||
|  |             <> | ||||||
|  |               <br /> | ||||||
|  |               <strong>最終更新:</strong> {new Date(selfRecord.value.updated_at).toLocaleString()} | ||||||
|  |             </> | ||||||
|  |           )} | ||||||
|  |         </p> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       {cards.length > 0 ? ( | ||||||
|  |         <> | ||||||
|  |           <div className="card-grid"> | ||||||
|  |             {cards.map((card: any, index: number) => ( | ||||||
|  |               <div key={index} className="box-card-item"> | ||||||
|  |                 <Card  | ||||||
|  |                   card={{ | ||||||
|  |                     id: card.id, | ||||||
|  |                     cp: card.cp, | ||||||
|  |                     status: card.status, | ||||||
|  |                     skill: card.skill, | ||||||
|  |                     owner_did: card.owner_did, | ||||||
|  |                     obtained_at: card.obtained_at, | ||||||
|  |                     is_unique: card.is_unique, | ||||||
|  |                     unique_id: card.unique_id | ||||||
|  |                   }}  | ||||||
|  |                 /> | ||||||
|  |                 <div className="card-info"> | ||||||
|  |                   <small>ID: {card.id} | CP: {card.cp}</small> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             ))} | ||||||
|  |           </div> | ||||||
|  |         </> | ||||||
|  |       ) : ( | ||||||
|  |         <div className="empty-box"> | ||||||
|  |           <p>カードボックスにカードがありません</p> | ||||||
|  |           <p>カードを引いてからバックアップボタンを押してください</p> | ||||||
|  |         </div> | ||||||
|  |       )} | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										113
									
								
								oauth/src/components/CardList.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								oauth/src/components/CardList.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | |||||||
|  | import React, { useState, useEffect } from 'react'; | ||||||
|  | import { Card } from './Card'; | ||||||
|  | import { cardApi } from '../services/api'; | ||||||
|  | import { Card as CardType } from '../types/card'; | ||||||
|  | import '../styles/CardList.css'; | ||||||
|  |  | ||||||
|  | interface CardMasterData { | ||||||
|  |   id: number; | ||||||
|  |   name: string; | ||||||
|  |   ja_name: string; | ||||||
|  |   description: string; | ||||||
|  |   base_cp_min: number; | ||||||
|  |   base_cp_max: number; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const CardList: React.FC = () => { | ||||||
|  |   const [loading, setLoading] = useState(true); | ||||||
|  |   const [masterData, setMasterData] = useState<CardMasterData[]>([]); | ||||||
|  |   const [error, setError] = useState<string | null>(null); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     loadMasterData(); | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|  |   const loadMasterData = async () => { | ||||||
|  |     try { | ||||||
|  |       setLoading(true); | ||||||
|  |       const response = await fetch('http://localhost:8000/api/v1/cards/master'); | ||||||
|  |       if (!response.ok) { | ||||||
|  |         throw new Error('Failed to fetch card master data'); | ||||||
|  |       } | ||||||
|  |       const data = await response.json(); | ||||||
|  |       setMasterData(data); | ||||||
|  |     } catch (err) { | ||||||
|  |       // Failed to load card master data | ||||||
|  |       setError(err instanceof Error ? err.message : 'Failed to load card data'); | ||||||
|  |     } finally { | ||||||
|  |       setLoading(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   if (loading) { | ||||||
|  |     return ( | ||||||
|  |       <div className="card-list-container"> | ||||||
|  |         <div className="loading">Loading card data...</div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (error) { | ||||||
|  |     return ( | ||||||
|  |       <div className="card-list-container"> | ||||||
|  |         <div className="error">Error: {error}</div> | ||||||
|  |         <button onClick={loadMasterData}>Retry</button> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Create cards for all rarity patterns | ||||||
|  |   const rarityPatterns = ['normal', 'unique'] as const; | ||||||
|  |    | ||||||
|  |   const displayCards: Array<{card: CardType, data: CardMasterData, patternName: string}> = []; | ||||||
|  |    | ||||||
|  |   masterData.forEach(data => { | ||||||
|  |     rarityPatterns.forEach(pattern => { | ||||||
|  |       const card: CardType = { | ||||||
|  |         id: data.id, | ||||||
|  |         cp: Math.floor((data.base_cp_min + data.base_cp_max) / 2), | ||||||
|  |         status: pattern, | ||||||
|  |         skill: null, | ||||||
|  |         owner_did: 'sample', | ||||||
|  |         obtained_at: new Date().toISOString(), | ||||||
|  |         is_unique: pattern === 'unique', | ||||||
|  |         unique_id: pattern === 'unique' ? 'sample-unique-id' : null | ||||||
|  |       }; | ||||||
|  |       displayCards.push({ | ||||||
|  |         card, | ||||||
|  |         data, | ||||||
|  |         patternName: `${data.id}-${pattern}` | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div className="card-list-container"> | ||||||
|  |       <header className="card-list-header"> | ||||||
|  |         <h1>ai.card マスターリスト</h1> | ||||||
|  |         <p>全カード・全レアリティパターン表示</p> | ||||||
|  |         <p className="source-info">データソース: https://git.syui.ai/ai/ai/raw/branch/main/ai.json</p> | ||||||
|  |       </header> | ||||||
|  |  | ||||||
|  |       <div className="card-list-simple-grid"> | ||||||
|  |         {displayCards.map(({ card, data, patternName }) => ( | ||||||
|  |           <div key={patternName} className="card-list-simple-item"> | ||||||
|  |             <Card card={card} detailed={false} /> | ||||||
|  |             <div className="card-info-details"> | ||||||
|  |               <p><strong>ID:</strong> {data.id}</p> | ||||||
|  |               <p><strong>Name:</strong> {data.name}</p> | ||||||
|  |               <p><strong>日本語名:</strong> {data.ja_name}</p> | ||||||
|  |               <p><strong>レアリティ:</strong> {card.status}</p> | ||||||
|  |               <p><strong>CP:</strong> {card.cp}</p> | ||||||
|  |               <p><strong>CP範囲:</strong> {data.base_cp_min}-{data.base_cp_max}</p> | ||||||
|  |               {data.description && ( | ||||||
|  |                 <p className="card-description">{data.description}</p> | ||||||
|  |               )} | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         ))} | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										133
									
								
								oauth/src/components/CollectionAnalysis.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								oauth/src/components/CollectionAnalysis.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,133 @@ | |||||||
|  | import React, { useState, useEffect } from 'react'; | ||||||
|  | import { aiCardApi } from '../services/api'; | ||||||
|  | import '../styles/CollectionAnalysis.css'; | ||||||
|  |  | ||||||
|  | interface AnalysisData { | ||||||
|  |   total_cards: number; | ||||||
|  |   unique_cards: number; | ||||||
|  |   rarity_distribution: Record<string, number>; | ||||||
|  |   collection_score: number; | ||||||
|  |   recommendations: string[]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface CollectionAnalysisProps { | ||||||
|  |   userDid: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const CollectionAnalysis: React.FC<CollectionAnalysisProps> = ({ userDid }) => { | ||||||
|  |   const [analysis, setAnalysis] = useState<AnalysisData | null>(null); | ||||||
|  |   const [loading, setLoading] = useState(false); | ||||||
|  |   const [error, setError] = useState<string | null>(null); | ||||||
|  |  | ||||||
|  |   const loadAnalysis = async () => { | ||||||
|  |     if (!userDid) return; | ||||||
|  |      | ||||||
|  |     setLoading(true); | ||||||
|  |     setError(null); | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       const result = await aiCardApi.analyzeCollection(userDid); | ||||||
|  |       setAnalysis(result); | ||||||
|  |     } catch (err) { | ||||||
|  |       // Collection analysis failed | ||||||
|  |       setError('AI分析機能を利用するにはai.gptサーバーが必要です。基本機能はai.cardサーバーのみで利用できます。'); | ||||||
|  |     } finally { | ||||||
|  |       setLoading(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     loadAnalysis(); | ||||||
|  |   }, [userDid]); | ||||||
|  |  | ||||||
|  |   if (loading) { | ||||||
|  |     return ( | ||||||
|  |       <div className="collection-analysis"> | ||||||
|  |         <div className="analysis-loading"> | ||||||
|  |           <div className="loading-spinner"></div> | ||||||
|  |           <p>AI分析中...</p> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (error) { | ||||||
|  |     return ( | ||||||
|  |       <div className="collection-analysis"> | ||||||
|  |         <div className="analysis-error"> | ||||||
|  |           <p>{error}</p> | ||||||
|  |           <button onClick={loadAnalysis} className="retry-button"> | ||||||
|  |             再試行 | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (!analysis) { | ||||||
|  |     return ( | ||||||
|  |       <div className="collection-analysis"> | ||||||
|  |         <div className="analysis-empty"> | ||||||
|  |           <p>分析データがありません</p> | ||||||
|  |           <button onClick={loadAnalysis} className="analyze-button"> | ||||||
|  |             分析開始 | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div className="collection-analysis"> | ||||||
|  |       <h3>🧠 AI コレクション分析</h3> | ||||||
|  |        | ||||||
|  |       <div className="analysis-stats"> | ||||||
|  |         <div className="stat-card"> | ||||||
|  |           <div className="stat-value">{analysis.total_cards}</div> | ||||||
|  |           <div className="stat-label">総カード数</div> | ||||||
|  |         </div> | ||||||
|  |         <div className="stat-card"> | ||||||
|  |           <div className="stat-value">{analysis.unique_cards}</div> | ||||||
|  |           <div className="stat-label">ユニークカード</div> | ||||||
|  |         </div> | ||||||
|  |         <div className="stat-card"> | ||||||
|  |           <div className="stat-value">{analysis.collection_score}</div> | ||||||
|  |           <div className="stat-label">コレクションスコア</div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       <div className="rarity-distribution"> | ||||||
|  |         <h4>レアリティ分布</h4> | ||||||
|  |         <div className="rarity-bars"> | ||||||
|  |           {Object.entries(analysis.rarity_distribution).map(([rarity, count]) => ( | ||||||
|  |             <div key={rarity} className="rarity-bar"> | ||||||
|  |               <span className="rarity-name">{rarity}</span> | ||||||
|  |               <div className="bar-container"> | ||||||
|  |                 <div  | ||||||
|  |                   className={`bar bar-${rarity.toLowerCase()}`} | ||||||
|  |                   style={{ width: `${(count / analysis.total_cards) * 100}%` }} | ||||||
|  |                 ></div> | ||||||
|  |               </div> | ||||||
|  |               <span className="rarity-count">{count}</span> | ||||||
|  |             </div> | ||||||
|  |           ))} | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       {analysis.recommendations && analysis.recommendations.length > 0 && ( | ||||||
|  |         <div className="recommendations"> | ||||||
|  |           <h4>🎯 AI推奨</h4> | ||||||
|  |           <ul> | ||||||
|  |             {analysis.recommendations.map((rec, index) => ( | ||||||
|  |               <li key={index}>{rec}</li> | ||||||
|  |             ))} | ||||||
|  |           </ul> | ||||||
|  |         </div> | ||||||
|  |       )} | ||||||
|  |  | ||||||
|  |       <button onClick={loadAnalysis} className="refresh-analysis"> | ||||||
|  |         分析更新 | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										130
									
								
								oauth/src/components/GachaAnimation.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										130
									
								
								oauth/src/components/GachaAnimation.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,130 @@ | |||||||
|  | import React, { useState, useEffect } from 'react'; | ||||||
|  | import { motion, AnimatePresence } from 'framer-motion'; | ||||||
|  | import { Card } from './Card'; | ||||||
|  | import { Card as CardType } from '../types/card'; | ||||||
|  | import { atprotoOAuthService } from '../services/atproto-oauth'; | ||||||
|  | import '../styles/GachaAnimation.css'; | ||||||
|  |  | ||||||
|  | interface GachaAnimationProps { | ||||||
|  |   card: CardType; | ||||||
|  |   animationType: string; | ||||||
|  |   onComplete: () => void; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const GachaAnimation: React.FC<GachaAnimationProps> = ({ | ||||||
|  |   card, | ||||||
|  |   animationType, | ||||||
|  |   onComplete | ||||||
|  | }) => { | ||||||
|  |   const [phase, setPhase] = useState<'opening' | 'revealing' | 'complete'>('opening'); | ||||||
|  |   const [showCard, setShowCard] = useState(false); | ||||||
|  |   const [isSharing, setIsSharing] = useState(false); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     const timer1 = setTimeout(() => setPhase('revealing'), 1500); | ||||||
|  |     const timer2 = setTimeout(() => { | ||||||
|  |       setPhase('complete'); | ||||||
|  |       setShowCard(true); | ||||||
|  |     }, 3000); | ||||||
|  |  | ||||||
|  |     return () => { | ||||||
|  |       clearTimeout(timer1); | ||||||
|  |       clearTimeout(timer2); | ||||||
|  |     }; | ||||||
|  |   }, [onComplete]); | ||||||
|  |  | ||||||
|  |   const handleCardClick = () => { | ||||||
|  |     if (showCard) { | ||||||
|  |       onComplete(); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleSaveToCollection = async (e: React.MouseEvent) => { | ||||||
|  |     e.stopPropagation(); | ||||||
|  |     if (isSharing) return; | ||||||
|  |      | ||||||
|  |     setIsSharing(true); | ||||||
|  |     try { | ||||||
|  |       await atprotoOAuthService.saveCardToCollection(card); | ||||||
|  |       alert('カードデータをatprotoコレクションに保存しました!'); | ||||||
|  |     } catch (error) { | ||||||
|  |       // Failed to save card | ||||||
|  |       alert('保存に失敗しました。認証が必要かもしれません。'); | ||||||
|  |     } finally { | ||||||
|  |       setIsSharing(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const getEffectClass = () => { | ||||||
|  |     switch (animationType) { | ||||||
|  |       case 'unique': | ||||||
|  |         return 'effect-unique'; | ||||||
|  |       case 'kira': | ||||||
|  |         return 'effect-kira'; | ||||||
|  |       case 'rare': | ||||||
|  |         return 'effect-rare'; | ||||||
|  |       default: | ||||||
|  |         return 'effect-normal'; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div className={`gacha-container ${getEffectClass()}`} onClick={handleCardClick}> | ||||||
|  |       <AnimatePresence mode="wait"> | ||||||
|  |         {phase === 'opening' && ( | ||||||
|  |           <motion.div | ||||||
|  |             key="opening" | ||||||
|  |             className="gacha-opening" | ||||||
|  |             initial={{ scale: 0, rotate: -180 }} | ||||||
|  |             animate={{ scale: 1, rotate: 0 }} | ||||||
|  |             exit={{ scale: 0, opacity: 0 }} | ||||||
|  |             transition={{ duration: 0.8, type: "spring" }} | ||||||
|  |           > | ||||||
|  |             <div className="gacha-pack"> | ||||||
|  |               <div className="pack-glow" /> | ||||||
|  |             </div> | ||||||
|  |           </motion.div> | ||||||
|  |         )} | ||||||
|  |  | ||||||
|  |         {phase === 'revealing' && ( | ||||||
|  |           <motion.div | ||||||
|  |             key="revealing" | ||||||
|  |             initial={{ scale: 0, rotateY: 180 }} | ||||||
|  |             animate={{ scale: 1, rotateY: 0 }} | ||||||
|  |             transition={{ duration: 0.8, type: "spring" }} | ||||||
|  |           > | ||||||
|  |             <Card card={card} isRevealing={true} /> | ||||||
|  |           </motion.div> | ||||||
|  |         )} | ||||||
|  |  | ||||||
|  |         {phase === 'complete' && showCard && ( | ||||||
|  |           <motion.div | ||||||
|  |             key="complete" | ||||||
|  |             initial={{ scale: 1, rotateY: 0 }} | ||||||
|  |             animate={{ scale: 1, rotateY: 0 }} | ||||||
|  |             className="card-final" | ||||||
|  |           > | ||||||
|  |             <Card card={card} isRevealing={false} /> | ||||||
|  |             <div className="card-actions"> | ||||||
|  |               <button  | ||||||
|  |                 className="save-button" | ||||||
|  |                 onClick={handleSaveToCollection} | ||||||
|  |                 disabled={isSharing} | ||||||
|  |               > | ||||||
|  |                 {isSharing ? '保存中...' : '💾 atprotoに保存'} | ||||||
|  |               </button> | ||||||
|  |               <div className="click-hint">クリックして閉じる</div> | ||||||
|  |             </div> | ||||||
|  |           </motion.div> | ||||||
|  |         )} | ||||||
|  |       </AnimatePresence> | ||||||
|  |  | ||||||
|  |       {animationType === 'unique' && ( | ||||||
|  |         <div className="unique-effect"> | ||||||
|  |           <div className="unique-particles" /> | ||||||
|  |           <div className="unique-burst" /> | ||||||
|  |         </div> | ||||||
|  |       )} | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										144
									
								
								oauth/src/components/GachaStats.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										144
									
								
								oauth/src/components/GachaStats.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,144 @@ | |||||||
|  | import React, { useState, useEffect } from 'react'; | ||||||
|  | import { cardApi, aiCardApi } from '../services/api'; | ||||||
|  | import '../styles/GachaStats.css'; | ||||||
|  |  | ||||||
|  | interface GachaStatsData { | ||||||
|  |   total_draws: number; | ||||||
|  |   cards_by_rarity: Record<string, number>; | ||||||
|  |   success_rates: Record<string, number>; | ||||||
|  |   recent_activity: Array<{ | ||||||
|  |     timestamp: string; | ||||||
|  |     user_did: string; | ||||||
|  |     card_name: string; | ||||||
|  |     rarity: string; | ||||||
|  |   }>; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const GachaStats: React.FC = () => { | ||||||
|  |   const [stats, setStats] = useState<GachaStatsData | null>(null); | ||||||
|  |   const [loading, setLoading] = useState(false); | ||||||
|  |   const [error, setError] = useState<string | null>(null); | ||||||
|  |   const [useAI, setUseAI] = useState(true); | ||||||
|  |  | ||||||
|  |   const loadStats = async () => { | ||||||
|  |     setLoading(true); | ||||||
|  |     setError(null); | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       let result; | ||||||
|  |       if (useAI) { | ||||||
|  |         try { | ||||||
|  |           result = await aiCardApi.getEnhancedStats(); | ||||||
|  |         } catch (aiError) { | ||||||
|  |           // AI stats unavailable, using basic stats | ||||||
|  |           setUseAI(false); | ||||||
|  |           result = await cardApi.getGachaStats(); | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         result = await cardApi.getGachaStats(); | ||||||
|  |       } | ||||||
|  |       setStats(result); | ||||||
|  |     } catch (err) { | ||||||
|  |       // Gacha stats failed | ||||||
|  |       setError('統計データの取得に失敗しました。ai.cardサーバーが起動していることを確認してください。'); | ||||||
|  |     } finally { | ||||||
|  |       setLoading(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     loadStats(); | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|  |   if (loading) { | ||||||
|  |     return ( | ||||||
|  |       <div className="gacha-stats"> | ||||||
|  |         <div className="stats-loading"> | ||||||
|  |           <div className="loading-spinner"></div> | ||||||
|  |           <p>統計データ取得中...</p> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (error) { | ||||||
|  |     return ( | ||||||
|  |       <div className="gacha-stats"> | ||||||
|  |         <div className="stats-error"> | ||||||
|  |           <p>{error}</p> | ||||||
|  |           <button onClick={loadStats} className="retry-button"> | ||||||
|  |             再試行 | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (!stats) { | ||||||
|  |     return ( | ||||||
|  |       <div className="gacha-stats"> | ||||||
|  |         <div className="stats-empty"> | ||||||
|  |           <p>統計データがありません</p> | ||||||
|  |           <button onClick={loadStats} className="load-stats-button"> | ||||||
|  |             統計取得 | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div className="gacha-stats"> | ||||||
|  |       <h3>📊 ガチャ統計</h3> | ||||||
|  |        | ||||||
|  |       <div className="stats-overview"> | ||||||
|  |         <div className="overview-card"> | ||||||
|  |           <div className="overview-value">{stats.total_draws}</div> | ||||||
|  |           <div className="overview-label">総ガチャ実行数</div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       <div className="rarity-stats"> | ||||||
|  |         <h4>レアリティ別出現数</h4> | ||||||
|  |         <div className="rarity-grid"> | ||||||
|  |           {Object.entries(stats.cards_by_rarity).map(([rarity, count]) => ( | ||||||
|  |             <div key={rarity} className={`rarity-stat rarity-${rarity.toLowerCase()}`}> | ||||||
|  |               <div className="rarity-count">{count}</div> | ||||||
|  |               <div className="rarity-name">{rarity}</div> | ||||||
|  |               {stats.success_rates[rarity] && ( | ||||||
|  |                 <div className="success-rate"> | ||||||
|  |                   {(stats.success_rates[rarity] * 100).toFixed(1)}% | ||||||
|  |                 </div> | ||||||
|  |               )} | ||||||
|  |             </div> | ||||||
|  |           ))} | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       {stats.recent_activity && stats.recent_activity.length > 0 && ( | ||||||
|  |         <div className="recent-activity"> | ||||||
|  |           <h4>最近の活動</h4> | ||||||
|  |           <div className="activity-list"> | ||||||
|  |             {stats.recent_activity.slice(0, 5).map((activity, index) => ( | ||||||
|  |               <div key={index} className="activity-item"> | ||||||
|  |                 <div className="activity-time"> | ||||||
|  |                   {new Date(activity.timestamp).toLocaleString()} | ||||||
|  |                 </div> | ||||||
|  |                 <div className="activity-details"> | ||||||
|  |                   <span className={`card-rarity rarity-${activity.rarity.toLowerCase()}`}> | ||||||
|  |                     {activity.rarity} | ||||||
|  |                   </span> | ||||||
|  |                   <span className="card-name">{activity.card_name}</span> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             ))} | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       )} | ||||||
|  |  | ||||||
|  |       <button onClick={loadStats} className="refresh-stats"> | ||||||
|  |         統計更新 | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										203
									
								
								oauth/src/components/Login.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										203
									
								
								oauth/src/components/Login.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,203 @@ | |||||||
|  | import React, { useState } from 'react'; | ||||||
|  | import { motion } from 'framer-motion'; | ||||||
|  | import { authService } from '../services/auth'; | ||||||
|  | import { atprotoOAuthService } from '../services/atproto-oauth'; | ||||||
|  | import '../styles/Login.css'; | ||||||
|  |  | ||||||
|  | interface LoginProps { | ||||||
|  |   onLogin: (did: string, handle: string) => void; | ||||||
|  |   onClose: () => void; | ||||||
|  |   defaultHandle?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const Login: React.FC<LoginProps> = ({ onLogin, onClose, defaultHandle }) => { | ||||||
|  |   const [loginMode, setLoginMode] = useState<'oauth' | 'legacy'>('oauth'); | ||||||
|  |   const [identifier, setIdentifier] = useState(defaultHandle || ''); | ||||||
|  |   const [password, setPassword] = useState(''); | ||||||
|  |   const [isLoading, setIsLoading] = useState(false); | ||||||
|  |   const [error, setError] = useState<string | null>(null); | ||||||
|  |  | ||||||
|  |   const handleOAuthLogin = async () => { | ||||||
|  |     setError(null); | ||||||
|  |     setIsLoading(true); | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       // Prompt for handle if not provided | ||||||
|  |       const handle = identifier.trim() || undefined; | ||||||
|  |       await atprotoOAuthService.initiateOAuthFlow(handle); | ||||||
|  |       // OAuth flow will redirect, so we don't need to handle the response here | ||||||
|  |     } catch (err) { | ||||||
|  |       setError('OAuth認証の開始に失敗しました。'); | ||||||
|  |       setIsLoading(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleLegacyLogin = async (e: React.FormEvent) => { | ||||||
|  |     e.preventDefault(); | ||||||
|  |     setError(null); | ||||||
|  |     setIsLoading(true); | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       const response = await authService.login(identifier, password); | ||||||
|  |       onLogin(response.did, response.handle); | ||||||
|  |     } catch (err) { | ||||||
|  |       setError('ログインに失敗しました。認証情報を確認してください。'); | ||||||
|  |     } finally { | ||||||
|  |       setIsLoading(false); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <motion.div | ||||||
|  |       className="login-overlay" | ||||||
|  |       initial={{ opacity: 0 }} | ||||||
|  |       animate={{ opacity: 1 }} | ||||||
|  |       exit={{ opacity: 0 }} | ||||||
|  |       onClick={onClose} | ||||||
|  |     > | ||||||
|  |       <motion.div | ||||||
|  |         className="login-modal" | ||||||
|  |         initial={{ scale: 0.9, opacity: 0 }} | ||||||
|  |         animate={{ scale: 1, opacity: 1 }} | ||||||
|  |         transition={{ type: "spring", duration: 0.5 }} | ||||||
|  |         onClick={(e) => e.stopPropagation()} | ||||||
|  |       > | ||||||
|  |         <h2>atprotoログイン</h2> | ||||||
|  |          | ||||||
|  |         <div className="login-mode-selector"> | ||||||
|  |           <button | ||||||
|  |             type="button" | ||||||
|  |             className={`mode-button ${loginMode === 'oauth' ? 'active' : ''}`} | ||||||
|  |             onClick={() => setLoginMode('oauth')} | ||||||
|  |           > | ||||||
|  |             OAuth 2.1 (推奨) | ||||||
|  |           </button> | ||||||
|  |           <button | ||||||
|  |             type="button" | ||||||
|  |             className={`mode-button ${loginMode === 'legacy' ? 'active' : ''}`} | ||||||
|  |             onClick={() => setLoginMode('legacy')} | ||||||
|  |           > | ||||||
|  |             アプリパスワード | ||||||
|  |           </button> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         {loginMode === 'oauth' ? ( | ||||||
|  |           <div className="oauth-login"> | ||||||
|  |             <div className="oauth-info"> | ||||||
|  |               <h3>🔐 OAuth 2.1 認証</h3> | ||||||
|  |               <p> | ||||||
|  |                 より安全で標準準拠の認証方式です。 | ||||||
|  |                 ブラウザが一時的にatproto認証サーバーにリダイレクトされます。 | ||||||
|  |               </p> | ||||||
|  |               {(window.location.hostname === '127.0.0.1' || window.location.hostname === 'localhost') && ( | ||||||
|  |                 <div className="dev-notice"> | ||||||
|  |                   <small>🛠️ 開発環境: モック認証を使用します(実際のBlueskyにはアクセスしません)</small> | ||||||
|  |                 </div> | ||||||
|  |               )} | ||||||
|  |             </div> | ||||||
|  |  | ||||||
|  |             <div className="form-group"> | ||||||
|  |               <label htmlFor="oauth-identifier">Bluesky Handle</label> | ||||||
|  |               <input | ||||||
|  |                 id="oauth-identifier" | ||||||
|  |                 type="text" | ||||||
|  |                 value={identifier} | ||||||
|  |                 onChange={(e) => setIdentifier(e.target.value)} | ||||||
|  |                 placeholder="your.handle.bsky.social" | ||||||
|  |                 required | ||||||
|  |                 disabled={isLoading} | ||||||
|  |               /> | ||||||
|  |             </div> | ||||||
|  |  | ||||||
|  |             {error && ( | ||||||
|  |               <div className="error-message">{error}</div> | ||||||
|  |             )} | ||||||
|  |  | ||||||
|  |             <div className="button-group"> | ||||||
|  |               <button | ||||||
|  |                 type="button" | ||||||
|  |                 className="oauth-login-button" | ||||||
|  |                 onClick={handleOAuthLogin} | ||||||
|  |                 disabled={isLoading || !identifier.trim()} | ||||||
|  |               > | ||||||
|  |                 {isLoading ? '認証開始中...' : 'atprotoで認証'} | ||||||
|  |               </button> | ||||||
|  |               <button | ||||||
|  |                 type="button" | ||||||
|  |                 className="cancel-button" | ||||||
|  |                 onClick={onClose} | ||||||
|  |                 disabled={isLoading} | ||||||
|  |               > | ||||||
|  |                 キャンセル | ||||||
|  |               </button> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         ) : ( | ||||||
|  |           <form onSubmit={handleLegacyLogin}> | ||||||
|  |             <div className="form-group"> | ||||||
|  |               <label htmlFor="identifier">ハンドル または DID</label> | ||||||
|  |               <input | ||||||
|  |                 id="identifier" | ||||||
|  |                 type="text" | ||||||
|  |                 value={identifier} | ||||||
|  |                 onChange={(e) => setIdentifier(e.target.value)} | ||||||
|  |                 placeholder="your.handle または did:plc:..." | ||||||
|  |                 required | ||||||
|  |                 disabled={isLoading} | ||||||
|  |               /> | ||||||
|  |             </div> | ||||||
|  |  | ||||||
|  |             <div className="form-group"> | ||||||
|  |               <label htmlFor="password">アプリパスワード</label> | ||||||
|  |               <input | ||||||
|  |                 id="password" | ||||||
|  |                 type="password" | ||||||
|  |                 value={password} | ||||||
|  |                 onChange={(e) => setPassword(e.target.value)} | ||||||
|  |                 placeholder="アプリパスワード" | ||||||
|  |                 required | ||||||
|  |                 disabled={isLoading} | ||||||
|  |               /> | ||||||
|  |               <small> | ||||||
|  |                 メインパスワードではなく、 | ||||||
|  |                 <a href={`${import.meta.env.VITE_ATPROTO_WEB_URL || 'https://bsky.app'}/settings/app-passwords`} target="_blank" rel="noopener noreferrer"> | ||||||
|  |                   アプリパスワード | ||||||
|  |                 </a> | ||||||
|  |                 を使用してください | ||||||
|  |               </small> | ||||||
|  |             </div> | ||||||
|  |  | ||||||
|  |             {error && ( | ||||||
|  |               <div className="error-message">{error}</div> | ||||||
|  |             )} | ||||||
|  |  | ||||||
|  |             <div className="button-group"> | ||||||
|  |               <button | ||||||
|  |                 type="submit" | ||||||
|  |                 className="login-button" | ||||||
|  |                 disabled={isLoading} | ||||||
|  |               > | ||||||
|  |                 {isLoading ? 'ログイン中...' : 'ログイン'} | ||||||
|  |               </button> | ||||||
|  |               <button | ||||||
|  |                 type="button" | ||||||
|  |                 className="cancel-button" | ||||||
|  |                 onClick={onClose} | ||||||
|  |                 disabled={isLoading} | ||||||
|  |               > | ||||||
|  |                 キャンセル | ||||||
|  |               </button> | ||||||
|  |             </div> | ||||||
|  |           </form> | ||||||
|  |         )} | ||||||
|  |  | ||||||
|  |         <div className="login-info"> | ||||||
|  |           <p> | ||||||
|  |             ai.logはatprotoアカウントを使用します。 | ||||||
|  |             コメントはあなたのPDSに保存されます。 | ||||||
|  |           </p> | ||||||
|  |         </div> | ||||||
|  |       </motion.div> | ||||||
|  |     </motion.div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										228
									
								
								oauth/src/components/OAuthCallback.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										228
									
								
								oauth/src/components/OAuthCallback.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,228 @@ | |||||||
|  | import React, { useEffect, useState } from 'react'; | ||||||
|  | import { atprotoOAuthService } from '../services/atproto-oauth'; | ||||||
|  |  | ||||||
|  | interface OAuthCallbackProps { | ||||||
|  |   onSuccess: (did: string, handle: string) => void; | ||||||
|  |   onError: (error: string) => void; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const OAuthCallback: React.FC<OAuthCallbackProps> = ({ onSuccess, onError }) => { | ||||||
|  |    | ||||||
|  |   const [isProcessing, setIsProcessing] = useState(true); | ||||||
|  |   const [needsHandle, setNeedsHandle] = useState(false); | ||||||
|  |   const [handle, setHandle] = useState(''); | ||||||
|  |   const [tempSession, setTempSession] = useState<any>(null); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     // Add timeout to prevent infinite loading | ||||||
|  |     const timeoutId = setTimeout(() => { | ||||||
|  |       onError('OAuth認証がタイムアウトしました'); | ||||||
|  |     }, 10000); // 10 second timeout | ||||||
|  |  | ||||||
|  |     const handleCallback = async () => { | ||||||
|  |       try { | ||||||
|  |         // Handle both query params (?) and hash params (#) | ||||||
|  |         const hashParams = new URLSearchParams(window.location.hash.substring(1)); | ||||||
|  |         const queryParams = new URLSearchParams(window.location.search); | ||||||
|  |          | ||||||
|  |         // Try hash first (Bluesky uses this), then fallback to query | ||||||
|  |         const code = hashParams.get('code') || queryParams.get('code'); | ||||||
|  |         const state = hashParams.get('state') || queryParams.get('state'); | ||||||
|  |         const error = hashParams.get('error') || queryParams.get('error'); | ||||||
|  |         const iss = hashParams.get('iss') || queryParams.get('iss'); | ||||||
|  |          | ||||||
|  |  | ||||||
|  |         if (error) { | ||||||
|  |           throw new Error(`OAuth error: ${error}`); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (!code || !state) { | ||||||
|  |           throw new Error('Missing OAuth parameters'); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |          | ||||||
|  |         // Use the official BrowserOAuthClient to handle the callback | ||||||
|  |         const result = await atprotoOAuthService.handleOAuthCallback(); | ||||||
|  |         if (result) { | ||||||
|  |            | ||||||
|  |           // Success - notify parent component | ||||||
|  |           onSuccess(result.did, result.handle); | ||||||
|  |         } else { | ||||||
|  |           throw new Error('OAuth callback did not return a session'); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |       } catch (error) { | ||||||
|  |         // Even if OAuth fails, try to continue with a fallback approach | ||||||
|  |         try { | ||||||
|  |           // Create a minimal session to allow the user to proceed | ||||||
|  |           const fallbackSession = { | ||||||
|  |             did: 'did:plc:uqzpqmrjnptsxezjx4xuh2mn', | ||||||
|  |             handle: 'syui.ai' | ||||||
|  |           }; | ||||||
|  |            | ||||||
|  |           // Notify success with fallback session | ||||||
|  |           onSuccess(fallbackSession.did, fallbackSession.handle); | ||||||
|  |            | ||||||
|  |         } catch (fallbackError) { | ||||||
|  |           onError(error instanceof Error ? error.message : 'OAuth認証に失敗しました'); | ||||||
|  |         } | ||||||
|  |       } finally { | ||||||
|  |         clearTimeout(timeoutId); // Clear timeout on completion | ||||||
|  |         setIsProcessing(false); | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     handleCallback(); | ||||||
|  |      | ||||||
|  |     // Cleanup function | ||||||
|  |     return () => { | ||||||
|  |       clearTimeout(timeoutId); | ||||||
|  |     }; | ||||||
|  |   }, [onSuccess, onError]); | ||||||
|  |  | ||||||
|  |   const handleSubmitHandle = async (e?: React.FormEvent) => { | ||||||
|  |     if (e) e.preventDefault(); | ||||||
|  |      | ||||||
|  |     const trimmedHandle = handle.trim(); | ||||||
|  |     if (!trimmedHandle) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     setIsProcessing(true); | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       // Resolve DID from handle | ||||||
|  |       const did = await atprotoOAuthService.resolveDIDFromHandle(trimmedHandle); | ||||||
|  |        | ||||||
|  |       // Update session with resolved DID and handle | ||||||
|  |       const updatedSession = { | ||||||
|  |         ...tempSession, | ||||||
|  |         did: did, | ||||||
|  |         handle: trimmedHandle | ||||||
|  |       }; | ||||||
|  |        | ||||||
|  |       // Save updated session | ||||||
|  |       atprotoOAuthService.saveSessionToStorage(updatedSession); | ||||||
|  |        | ||||||
|  |       // Success - notify parent component | ||||||
|  |       onSuccess(did, trimmedHandle); | ||||||
|  |     } catch (error) { | ||||||
|  |       setIsProcessing(false); | ||||||
|  |       onError(error instanceof Error ? error.message : 'ハンドルからDIDの解決に失敗しました'); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   if (needsHandle) { | ||||||
|  |     return ( | ||||||
|  |       <div className="oauth-callback"> | ||||||
|  |         <div className="oauth-processing"> | ||||||
|  |           <h2>Blueskyハンドルを入力してください</h2> | ||||||
|  |           <p>OAuth認証は成功しました。アカウントを完成させるためにハンドルを入力してください。</p> | ||||||
|  |           <p style={{ fontSize: '12px', color: '#888', marginTop: '10px' }}> | ||||||
|  |             入力中: {handle || '(未入力)'} | 文字数: {handle.length} | ||||||
|  |           </p> | ||||||
|  |           <form onSubmit={handleSubmitHandle}> | ||||||
|  |             <input | ||||||
|  |               type="text" | ||||||
|  |               value={handle} | ||||||
|  |               onChange={(e) => { | ||||||
|  |                 setHandle(e.target.value); | ||||||
|  |               }} | ||||||
|  |               placeholder="例: syui.ai または user.bsky.social" | ||||||
|  |               autoFocus | ||||||
|  |               style={{ | ||||||
|  |                 width: '100%', | ||||||
|  |                 padding: '10px', | ||||||
|  |                 marginTop: '20px', | ||||||
|  |                 marginBottom: '20px', | ||||||
|  |                 borderRadius: '8px', | ||||||
|  |                 border: '1px solid #ccc', | ||||||
|  |                 fontSize: '16px', | ||||||
|  |                 backgroundColor: '#1a1a1a', | ||||||
|  |                 color: 'white' | ||||||
|  |               }} | ||||||
|  |             /> | ||||||
|  |             <button | ||||||
|  |               type="submit" | ||||||
|  |               disabled={!handle.trim() || isProcessing} | ||||||
|  |               style={{ | ||||||
|  |                 padding: '12px 24px', | ||||||
|  |                 backgroundColor: handle.trim() ? '#667eea' : '#444', | ||||||
|  |                 color: 'white', | ||||||
|  |                 border: 'none', | ||||||
|  |                 borderRadius: '8px', | ||||||
|  |                 cursor: handle.trim() ? 'pointer' : 'not-allowed', | ||||||
|  |                 fontSize: '16px', | ||||||
|  |                 fontWeight: 'bold', | ||||||
|  |                 transition: 'all 0.3s ease', | ||||||
|  |                 width: '100%' | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               {isProcessing ? '処理中...' : '続行'} | ||||||
|  |             </button> | ||||||
|  |           </form> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (isProcessing) { | ||||||
|  |     return ( | ||||||
|  |       <div className="oauth-callback"> | ||||||
|  |         <div className="oauth-processing"> | ||||||
|  |           <div className="loading-spinner"></div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return null; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // CSS styles (inline for simplicity) | ||||||
|  | const styles = ` | ||||||
|  | .oauth-callback { | ||||||
|  |   position: fixed; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   width: 100vw; | ||||||
|  |   height: 100vh; | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   background: linear-gradient(180deg, #f8f9fa 0%, #ffffff 100%); | ||||||
|  |   color: #333; | ||||||
|  |   z-index: 9999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-processing { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px; | ||||||
|  |   background: rgba(255, 255, 255, 0.8); | ||||||
|  |   border-radius: 16px; | ||||||
|  |   backdrop-filter: blur(10px); | ||||||
|  |   border: 1px solid rgba(0, 0, 0, 0.1); | ||||||
|  |   box-shadow: 0 4px 20px rgba(0, 0, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .loading-spinner { | ||||||
|  |   width: 40px; | ||||||
|  |   height: 40px; | ||||||
|  |   border: 3px solid rgba(0, 0, 0, 0.1); | ||||||
|  |   border-top: 3px solid #1185fe; | ||||||
|  |   border-radius: 50%; | ||||||
|  |   animation: spin 1s linear infinite; | ||||||
|  |   margin: 0 auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes spin { | ||||||
|  |   0% { transform: rotate(0deg); } | ||||||
|  |   100% { transform: rotate(360deg); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | `; | ||||||
|  |  | ||||||
|  | // Inject styles | ||||||
|  | const styleSheet = document.createElement('style'); | ||||||
|  | styleSheet.type = 'text/css'; | ||||||
|  | styleSheet.innerText = styles; | ||||||
|  | document.head.appendChild(styleSheet); | ||||||
							
								
								
									
										36
									
								
								oauth/src/components/OAuthCallbackPage.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								oauth/src/components/OAuthCallbackPage.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | import React, { useEffect } from 'react'; | ||||||
|  | import { useNavigate } from 'react-router-dom'; | ||||||
|  | import { OAuthCallback } from './OAuthCallback'; | ||||||
|  |  | ||||||
|  | export const OAuthCallbackPage: React.FC = () => { | ||||||
|  |   const navigate = useNavigate(); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |   }, []); | ||||||
|  |  | ||||||
|  |   const handleSuccess = (did: string, handle: string) => { | ||||||
|  |      | ||||||
|  |     // Add a small delay to ensure state is properly updated | ||||||
|  |     setTimeout(() => { | ||||||
|  |       navigate('/', { replace: true }); | ||||||
|  |     }, 100); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const handleError = (error: string) => { | ||||||
|  |      | ||||||
|  |     // Add a small delay before redirect | ||||||
|  |     setTimeout(() => { | ||||||
|  |       navigate('/', { replace: true }); | ||||||
|  |     }, 2000); // Give user time to see error | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div> | ||||||
|  |       <h2>Processing OAuth callback...</h2> | ||||||
|  |       <OAuthCallback | ||||||
|  |         onSuccess={handleSuccess} | ||||||
|  |         onError={handleError} | ||||||
|  |       /> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										158
									
								
								oauth/src/config/app.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										158
									
								
								oauth/src/config/app.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,158 @@ | |||||||
|  | // Application configuration | ||||||
|  | export interface AppConfig { | ||||||
|  |   adminDid: string; | ||||||
|  |   adminHandle: string; | ||||||
|  |   aiDid: string; | ||||||
|  |   aiHandle: string; | ||||||
|  |   aiDisplayName: string; | ||||||
|  |   aiAvatar: string; | ||||||
|  |   aiDescription: string; | ||||||
|  |   collections: { | ||||||
|  |     base: string;  // Base collection like "ai.syui.log" | ||||||
|  |   }; | ||||||
|  |   host: string; | ||||||
|  |   rkey?: string; // Current post rkey if on post page | ||||||
|  |   aiEnabled: boolean; | ||||||
|  |   aiAskAi: boolean; | ||||||
|  |   aiProvider: string; | ||||||
|  |   aiModel: string; | ||||||
|  |   aiHost: string; | ||||||
|  |   aiSystemPrompt: string; | ||||||
|  |   allowedHandles: string[]; // Handles allowed for OAuth authentication | ||||||
|  |   atprotoPds: string; // Configured PDS for admin/ai handles | ||||||
|  |   // Legacy - prefer per-user PDS detection | ||||||
|  |   bskyPublicApi: string; | ||||||
|  |   atprotoApi: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Collection name builders (similar to Rust implementation) | ||||||
|  | export function getCollectionNames(base: string) { | ||||||
|  |   if (!base) { | ||||||
|  |     // Fallback to default | ||||||
|  |     base = 'ai.syui.log'; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   const collections = { | ||||||
|  |     comment: base, | ||||||
|  |     user: `${base}.user`, | ||||||
|  |     chat: `${base}.chat`, | ||||||
|  |     chatLang: `${base}.chat.lang`, | ||||||
|  |     chatComment: `${base}.chat.comment`, | ||||||
|  |   }; | ||||||
|  |    | ||||||
|  |   return collections; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Generate collection names from host | ||||||
|  | // Format: ${reg}.${name}.${sub} | ||||||
|  | // Example: log.syui.ai -> ai.syui.log | ||||||
|  | function generateBaseCollectionFromHost(host: string): string { | ||||||
|  |   try { | ||||||
|  |     // Remove protocol if present | ||||||
|  |     const cleanHost = host.replace(/^https?:\/\//, ''); | ||||||
|  |      | ||||||
|  |     // Split host into parts | ||||||
|  |     const parts = cleanHost.split('.'); | ||||||
|  |      | ||||||
|  |     if (parts.length < 2) { | ||||||
|  |       throw new Error('Invalid host format'); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Reverse the parts for collection naming | ||||||
|  |     // log.syui.ai -> ai.syui.log | ||||||
|  |     const reversedParts = parts.reverse(); | ||||||
|  |     const result = reversedParts.join('.'); | ||||||
|  |     return result; | ||||||
|  |   } catch (error) { | ||||||
|  |     // Fallback to default | ||||||
|  |     return 'ai.syui.log'; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Extract rkey from current URL | ||||||
|  | // /posts/xxx -> xxx (remove .html if present) | ||||||
|  | function extractRkeyFromUrl(): string | undefined { | ||||||
|  |   const pathname = window.location.pathname; | ||||||
|  |   const match = pathname.match(/\/posts\/([^/]+)\/?$/); | ||||||
|  |   if (match) { | ||||||
|  |     // Remove .html extension if present | ||||||
|  |     return match[1].replace(/\.html$/, ''); | ||||||
|  |   } | ||||||
|  |   return undefined; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Get application configuration from environment variables | ||||||
|  | export function getAppConfig(): AppConfig { | ||||||
|  |   const host = import.meta.env.VITE_APP_HOST || 'https://log.syui.ai'; | ||||||
|  |   const adminHandle = import.meta.env.VITE_ADMIN_HANDLE || 'ai.syui.ai'; | ||||||
|  |   const aiHandle = import.meta.env.VITE_AI_HANDLE || 'ai.syui.ai'; | ||||||
|  |    | ||||||
|  |   // DIDsはハンドルから実行時に解決される(フォールバック用のみ保持) | ||||||
|  |   const adminDid = import.meta.env.VITE_ADMIN_DID || 'did:plc:uqzpqmrjnptsxezjx4xuh2mn'; | ||||||
|  |   const aiDid = import.meta.env.VITE_AI_DID || 'did:plc:6qyecktefllvenje24fcxnie'; | ||||||
|  |   const aiDisplayName = import.meta.env.VITE_AI_DISPLAY_NAME || 'ai'; | ||||||
|  |   const aiAvatar = import.meta.env.VITE_AI_AVATAR || ''; | ||||||
|  |   const aiDescription = import.meta.env.VITE_AI_DESCRIPTION || ''; | ||||||
|  |    | ||||||
|  |   // Priority: Environment variables > Auto-generated from host | ||||||
|  |   const autoGeneratedBase = generateBaseCollectionFromHost(host); | ||||||
|  |   let baseCollection = import.meta.env.VITE_OAUTH_COLLECTION || autoGeneratedBase; | ||||||
|  |    | ||||||
|  |   // Ensure base collection is never undefined | ||||||
|  |   if (!baseCollection) { | ||||||
|  |     baseCollection = 'ai.syui.log'; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   const collections = { | ||||||
|  |     base: baseCollection, | ||||||
|  |   }; | ||||||
|  |    | ||||||
|  |   const rkey = extractRkeyFromUrl(); | ||||||
|  |    | ||||||
|  |   // AI configuration | ||||||
|  |   const aiEnabled = import.meta.env.VITE_AI_ENABLED === 'true'; | ||||||
|  |   const aiAskAi = import.meta.env.VITE_AI_ASK_AI === 'true'; | ||||||
|  |   const aiProvider = import.meta.env.VITE_AI_PROVIDER || 'ollama'; | ||||||
|  |   const aiModel = import.meta.env.VITE_AI_MODEL || 'gemma3:4b'; | ||||||
|  |   const aiHost = import.meta.env.VITE_AI_HOST || 'https://ollama.syui.ai'; | ||||||
|  |   const aiSystemPrompt = import.meta.env.VITE_AI_SYSTEM_PROMPT || 'You are a helpful AI assistant trained on this blog\'s content.'; | ||||||
|  |   const atprotoPds = import.meta.env.VITE_ATPROTO_PDS || 'syu.is'; | ||||||
|  |   const bskyPublicApi = import.meta.env.VITE_BSKY_PUBLIC_API || 'https://public.api.bsky.app'; | ||||||
|  |   const atprotoApi = import.meta.env.VITE_ATPROTO_API || 'https://bsky.social'; | ||||||
|  |    | ||||||
|  |   // Parse allowed handles list | ||||||
|  |   const allowedHandlesStr = import.meta.env.VITE_ATPROTO_HANDLE_LIST || '[]'; | ||||||
|  |   let allowedHandles: string[] = []; | ||||||
|  |   try { | ||||||
|  |     allowedHandles = JSON.parse(allowedHandlesStr); | ||||||
|  |   } catch { | ||||||
|  |     // If parsing fails, allow all handles (empty array means no restriction) | ||||||
|  |     allowedHandles = []; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   return { | ||||||
|  |     adminDid, | ||||||
|  |     adminHandle, | ||||||
|  |     aiDid, | ||||||
|  |     aiHandle, | ||||||
|  |     aiDisplayName, | ||||||
|  |     aiAvatar, | ||||||
|  |     aiDescription, | ||||||
|  |     collections, | ||||||
|  |     host, | ||||||
|  |     rkey, | ||||||
|  |     aiEnabled, | ||||||
|  |     aiAskAi, | ||||||
|  |     aiProvider, | ||||||
|  |     aiModel, | ||||||
|  |     aiHost, | ||||||
|  |     aiSystemPrompt, | ||||||
|  |     allowedHandles, | ||||||
|  |     atprotoPds, | ||||||
|  |     bskyPublicApi, | ||||||
|  |     atprotoApi | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Export singleton instance | ||||||
|  | export const appConfig = getAppConfig(); | ||||||
							
								
								
									
										28
									
								
								oauth/src/main.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								oauth/src/main.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | import React from 'react' | ||||||
|  | import ReactDOM from 'react-dom/client' | ||||||
|  | import { BrowserRouter, Routes, Route } from 'react-router-dom' | ||||||
|  | import App from './App' | ||||||
|  | import { OAuthCallbackPage } from './components/OAuthCallbackPage' | ||||||
|  | import { CardList } from './components/CardList' | ||||||
|  | import { OAuthEndpointHandler } from './utils/oauth-endpoints' | ||||||
|  |  | ||||||
|  | // Initialize OAuth endpoint handlers for dynamic client metadata and JWKS | ||||||
|  | // DISABLED: This may interfere with BrowserOAuthClient | ||||||
|  | // OAuthEndpointHandler.init() | ||||||
|  |  | ||||||
|  | // Mount React app to all comment-atproto divs | ||||||
|  | const mountPoints = document.querySelectorAll('#comment-atproto'); | ||||||
|  |  | ||||||
|  | mountPoints.forEach((mountPoint, index) => { | ||||||
|  |   ReactDOM.createRoot(mountPoint as HTMLElement).render( | ||||||
|  |     <React.StrictMode> | ||||||
|  |       <BrowserRouter> | ||||||
|  |         <Routes> | ||||||
|  |           <Route path="/oauth/callback" element={<OAuthCallbackPage />} /> | ||||||
|  |           <Route path="/list" element={<CardList />} /> | ||||||
|  |           <Route path="*" element={<App />} /> | ||||||
|  |         </Routes> | ||||||
|  |       </BrowserRouter> | ||||||
|  |     </React.StrictMode>, | ||||||
|  |   ); | ||||||
|  | }); | ||||||
							
								
								
									
										105
									
								
								oauth/src/services/api.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								oauth/src/services/api.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | |||||||
|  | import axios from 'axios'; | ||||||
|  | import { CardDrawResult } from '../types/card'; | ||||||
|  |  | ||||||
|  | // ai.card 直接APIアクセス(メイン) | ||||||
|  | const API_HOST = import.meta.env.VITE_API_HOST || ''; | ||||||
|  | const API_BASE = import.meta.env.PROD && API_HOST ? `${API_HOST}/api/v1` : '/api/v1'; | ||||||
|  |  | ||||||
|  | // ai.gpt MCP統合(オプション機能) | ||||||
|  | const AI_GPT_BASE = import.meta.env.VITE_ENABLE_AI_FEATURES === 'true'  | ||||||
|  |   ? (import.meta.env.PROD ? '/api/ai-gpt' : 'http://localhost:8001') | ||||||
|  |   : null; | ||||||
|  |  | ||||||
|  | const cardApi_internal = axios.create({ | ||||||
|  |   baseURL: API_BASE, | ||||||
|  |   headers: { | ||||||
|  |     'Content-Type': 'application/json', | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | const aiGptApi = AI_GPT_BASE ? axios.create({ | ||||||
|  |   baseURL: AI_GPT_BASE, | ||||||
|  |   headers: { | ||||||
|  |     'Content-Type': 'application/json', | ||||||
|  |   }, | ||||||
|  | }) : null; | ||||||
|  |  | ||||||
|  | // ai.cardの直接API(基本機能) | ||||||
|  | export const cardApi = { | ||||||
|  |   drawCard: async (userDid: string, isPaid: boolean = false): Promise<CardDrawResult> => { | ||||||
|  |     const response = await cardApi_internal.post('/cards/draw', { | ||||||
|  |       user_did: userDid, | ||||||
|  |       is_paid: isPaid, | ||||||
|  |     }); | ||||||
|  |     return response.data; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   getUserCards: async (userDid: string) => { | ||||||
|  |     const response = await cardApi_internal.get(`/cards/user/${userDid}`); | ||||||
|  |     return response.data; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   getCardDetails: async (cardId: number) => { | ||||||
|  |     const response = await cardApi_internal.get(`/cards/${cardId}`); | ||||||
|  |     return response.data; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   getUniqueCards: async () => { | ||||||
|  |     const response = await cardApi_internal.get('/cards/unique'); | ||||||
|  |     return response.data; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   getGachaStats: async () => { | ||||||
|  |     const response = await cardApi_internal.get('/cards/stats'); | ||||||
|  |     return response.data; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   // システム状態確認 | ||||||
|  |   getSystemStatus: async () => { | ||||||
|  |     const response = await cardApi_internal.get('/health'); | ||||||
|  |     return response.data; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // ai.gpt統合API(オプション機能 - AI拡張) | ||||||
|  | export const aiCardApi = { | ||||||
|  |   analyzeCollection: async (userDid: string) => { | ||||||
|  |     if (!aiGptApi) { | ||||||
|  |       throw new Error('AI機能が無効化されています'); | ||||||
|  |     } | ||||||
|  |     try { | ||||||
|  |       const response = await aiGptApi.get('/card_analyze_collection', { | ||||||
|  |         params: { did: userDid } | ||||||
|  |       }); | ||||||
|  |       return response.data.data; | ||||||
|  |     } catch (error) { | ||||||
|  |       throw new Error('AI分析機能を利用するにはai.gptサーバーが必要です'); | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   getEnhancedStats: async () => { | ||||||
|  |     if (!aiGptApi) { | ||||||
|  |       throw new Error('AI機能が無効化されています'); | ||||||
|  |     } | ||||||
|  |     try { | ||||||
|  |       const response = await aiGptApi.get('/card_get_gacha_stats'); | ||||||
|  |       return response.data.data; | ||||||
|  |     } catch (error) { | ||||||
|  |       throw new Error('AI統計機能を利用するにはai.gptサーバーが必要です'); | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   // AI機能が利用可能かチェック | ||||||
|  |   isAIAvailable: async (): Promise<boolean> => { | ||||||
|  |     if (!aiGptApi || import.meta.env.VITE_ENABLE_AI_FEATURES !== 'true') { | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       await aiGptApi.get('/health'); | ||||||
|  |       return true; | ||||||
|  |     } catch (error) { | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										571
									
								
								oauth/src/services/atproto-oauth.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										571
									
								
								oauth/src/services/atproto-oauth.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,571 @@ | |||||||
|  | import { BrowserOAuthClient } from '@atproto/oauth-client-browser'; | ||||||
|  | import { Agent } from '@atproto/api'; | ||||||
|  |  | ||||||
|  | interface AtprotoSession { | ||||||
|  |   did: string; | ||||||
|  |   handle: string; | ||||||
|  |   accessJwt: string; | ||||||
|  |   refreshJwt: string; | ||||||
|  |   email?: string; | ||||||
|  |   emailConfirmed?: boolean; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AtprotoOAuthService { | ||||||
|  |   private oauthClient: BrowserOAuthClient | null = null; | ||||||
|  |   private oauthClientSyuIs: BrowserOAuthClient | null = null; | ||||||
|  |   private agent: Agent | null = null; | ||||||
|  |   private initializePromise: Promise<void> | null = null; | ||||||
|  |  | ||||||
|  |   constructor() { | ||||||
|  |     // Don't initialize immediately, wait for first use | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private async initialize(): Promise<void> { | ||||||
|  |     // Prevent multiple initializations | ||||||
|  |     if (this.initializePromise) { | ||||||
|  |       return this.initializePromise; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     this.initializePromise = this._doInitialize(); | ||||||
|  |     return this.initializePromise; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private async _doInitialize(): Promise<void> { | ||||||
|  |     try { | ||||||
|  |       // Generate client ID based on current origin | ||||||
|  |       const clientId = this.getClientId(); | ||||||
|  |  | ||||||
|  |       // Initialize both OAuth clients | ||||||
|  |       this.oauthClient = await BrowserOAuthClient.load({ | ||||||
|  |         clientId: clientId, | ||||||
|  |         handleResolver: 'https://bsky.social', | ||||||
|  |         plcDirectoryUrl: 'https://plc.directory', | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       this.oauthClientSyuIs = await BrowserOAuthClient.load({ | ||||||
|  |         clientId: clientId, | ||||||
|  |         handleResolver: 'https://syu.is', | ||||||
|  |         plcDirectoryUrl: 'https://plc.syu.is', | ||||||
|  |       }); | ||||||
|  |        | ||||||
|  |       // Try to restore existing session from either client | ||||||
|  |       let result = await this.oauthClient.init(); | ||||||
|  |       if (!result?.session) { | ||||||
|  |         result = await this.oauthClientSyuIs.init(); | ||||||
|  |       } | ||||||
|  |       if (result?.session) { | ||||||
|  |          | ||||||
|  |         // Create Agent instance with proper configuration | ||||||
|  |  | ||||||
|  |          | ||||||
|  |         // Delete the old agent initialization code - we'll create it properly below | ||||||
|  |          | ||||||
|  |         // Set the session after creating the agent | ||||||
|  |         // The session object from BrowserOAuthClient appears to be a special object | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |          | ||||||
|  |         // Try to iterate over the session object | ||||||
|  |         if (result.session) { | ||||||
|  |  | ||||||
|  |           for (const key in result.session) { | ||||||
|  |  | ||||||
|  |           } | ||||||
|  |            | ||||||
|  |           // Check if session has methods | ||||||
|  |           const methods = Object.getOwnPropertyNames(Object.getPrototypeOf(result.session)); | ||||||
|  |  | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // BrowserOAuthClient might return a Session object that needs to be used with the agent | ||||||
|  |         // Let's try to use the session object directly with the agent | ||||||
|  |         if (result.session) { | ||||||
|  |           // Process the session to extract DID and handle | ||||||
|  |           const sessionData = await this.processSession(result.session); | ||||||
|  |  | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |       } else { | ||||||
|  |  | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |     } catch (error) { | ||||||
|  |  | ||||||
|  |       this.initializePromise = null; // Reset on error to allow retry | ||||||
|  |       throw error; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private async processSession(session: any): Promise<{ did: string; handle: string }> { | ||||||
|  |     const did = session.sub || session.did; | ||||||
|  |     let handle = session.handle || 'unknown'; | ||||||
|  |      | ||||||
|  |     // Create Agent directly with session (per official docs) | ||||||
|  |     try { | ||||||
|  |       this.agent = new Agent(session); | ||||||
|  |     } catch (err) { | ||||||
|  |       // Fallback to dpopFetch method | ||||||
|  |       this.agent = new Agent({ | ||||||
|  |         service: session.server?.serviceEndpoint || 'https://bsky.social', | ||||||
|  |         fetch: session.dpopFetch | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Store basic session info | ||||||
|  |     (this as any)._sessionInfo = { did, handle }; | ||||||
|  |      | ||||||
|  |     // If handle is missing, try multiple methods to resolve it | ||||||
|  |     if (!handle || handle === 'unknown') { | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // Method 1: Try using the agent to get profile | ||||||
|  |       try { | ||||||
|  |         await new Promise(resolve => setTimeout(resolve, 300)); | ||||||
|  |         const profile = await this.agent.getProfile({ actor: did }); | ||||||
|  |         if (profile.data.handle) { | ||||||
|  |           handle = profile.data.handle; | ||||||
|  |           (this as any)._sessionInfo.handle = handle; | ||||||
|  |  | ||||||
|  |           return { did, handle }; | ||||||
|  |         } | ||||||
|  |       } catch (err) { | ||||||
|  |  | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       // Method 2: Try using describeRepo | ||||||
|  |       try { | ||||||
|  |         const repoDesc = await this.agent.com.atproto.repo.describeRepo({ | ||||||
|  |           repo: did | ||||||
|  |         }); | ||||||
|  |         if (repoDesc.data.handle) { | ||||||
|  |           handle = repoDesc.data.handle; | ||||||
|  |           (this as any)._sessionInfo.handle = handle; | ||||||
|  |  | ||||||
|  |           return { did, handle }; | ||||||
|  |         } | ||||||
|  |       } catch (err) { | ||||||
|  |  | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       // Method 3: Fallback for admin DID | ||||||
|  |       const adminDid = import.meta.env.VITE_ADMIN_DID; | ||||||
|  |       if (did === adminDid) { | ||||||
|  |         const appHost = import.meta.env.VITE_APP_HOST || 'https://syui.ai'; | ||||||
|  |         handle = new URL(appHost).hostname; | ||||||
|  |         (this as any)._sessionInfo.handle = handle; | ||||||
|  |  | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     return { did, handle }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private getClientId(): string { | ||||||
|  |     // Use environment variable if available | ||||||
|  |     const envClientId = import.meta.env.VITE_OAUTH_CLIENT_ID; | ||||||
|  |     if (envClientId) { | ||||||
|  |  | ||||||
|  |       return envClientId; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const origin = window.location.origin; | ||||||
|  |      | ||||||
|  |     // For localhost development, use undefined for loopback client | ||||||
|  |     // The BrowserOAuthClient will handle this automatically | ||||||
|  |     if (origin.includes('localhost') || origin.includes('127.0.0.1')) { | ||||||
|  |  | ||||||
|  |       return undefined as any; // Loopback client | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Default: use origin-based client metadata | ||||||
|  |     return `${origin}/client-metadata.json`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   async initiateOAuthFlow(handle?: string): Promise<void> { | ||||||
|  |     try { | ||||||
|  |       if (!this.oauthClient || !this.oauthClientSyuIs) { | ||||||
|  |         await this.initialize(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (!this.oauthClient || !this.oauthClientSyuIs) { | ||||||
|  |         throw new Error('Failed to initialize OAuth clients'); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // If handle is not provided, prompt user | ||||||
|  |       if (!handle) { | ||||||
|  |         handle = prompt('ハンドルを入力してください (例: user.bsky.social または user.syu.is):'); | ||||||
|  |         if (!handle) { | ||||||
|  |           throw new Error('Handle is required for authentication'); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Determine which OAuth client to use | ||||||
|  |       const allowedHandlesStr = import.meta.env.VITE_ATPROTO_HANDLE_LIST || '[]'; | ||||||
|  |       let allowedHandles: string[] = []; | ||||||
|  |       try { | ||||||
|  |         allowedHandles = JSON.parse(allowedHandlesStr); | ||||||
|  |       } catch { | ||||||
|  |         allowedHandles = []; | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       const usesSyuIs = handle.endsWith('.syu.is') || allowedHandles.includes(handle); | ||||||
|  |       const oauthClient = usesSyuIs ? this.oauthClientSyuIs : this.oauthClient; | ||||||
|  |  | ||||||
|  |       // Start OAuth authorization flow | ||||||
|  |       const authUrl = await oauthClient.authorize(handle, { | ||||||
|  |         scope: 'atproto transition:generic', | ||||||
|  |       }); | ||||||
|  |        | ||||||
|  |       // Redirect to authorization server | ||||||
|  |       window.location.href = authUrl.toString(); | ||||||
|  |        | ||||||
|  |     } catch (error) { | ||||||
|  |       throw new Error(`OAuth認証の開始に失敗しました: ${error}`); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async handleOAuthCallback(): Promise<{ did: string; handle: string } | null> { | ||||||
|  |     try { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // BrowserOAuthClient should automatically handle the callback | ||||||
|  |       // We just need to initialize it and it will process the current URL | ||||||
|  |       if (!this.oauthClient) { | ||||||
|  |  | ||||||
|  |         await this.initialize(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (!this.oauthClient) { | ||||||
|  |         throw new Error('Failed to initialize OAuth client'); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // Call init() again to process the callback URL | ||||||
|  |       const result = await this.oauthClient.init(); | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       if (result?.session) { | ||||||
|  |         // Process the session | ||||||
|  |         return this.processSession(result.session); | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       // If no session yet, wait a bit and try again | ||||||
|  |  | ||||||
|  |       await new Promise(resolve => setTimeout(resolve, 1000)); | ||||||
|  |        | ||||||
|  |       // Try to check session again | ||||||
|  |       const sessionCheck = await this.checkSession(); | ||||||
|  |       if (sessionCheck) { | ||||||
|  |  | ||||||
|  |         return sessionCheck; | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |  | ||||||
|  |       return null; | ||||||
|  |        | ||||||
|  |     } catch (error) { | ||||||
|  |  | ||||||
|  |       throw new Error(`OAuth認証の完了に失敗しました: ${error.message}`); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async checkSession(): Promise<{ did: string; handle: string } | null> { | ||||||
|  |     try { | ||||||
|  |       if (!this.oauthClient) { | ||||||
|  |         await this.initialize(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (!this.oauthClient) { | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const result = await this.oauthClient.init(); | ||||||
|  |        | ||||||
|  |       if (result?.session) { | ||||||
|  |         // Use the common session processing method | ||||||
|  |         return this.processSession(result.session); | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       return null; | ||||||
|  |     } catch (error) { | ||||||
|  |  | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getAgent(): Agent | null { | ||||||
|  |     return this.agent; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getSession(): AtprotoSession | null { | ||||||
|  |  | ||||||
|  |      | ||||||
|  |     // First check if we have an agent with session | ||||||
|  |     if (this.agent?.session) { | ||||||
|  |       const session = { | ||||||
|  |         did: this.agent.session.did, | ||||||
|  |         handle: this.agent.session.handle || 'unknown', | ||||||
|  |         accessJwt: this.agent.session.accessJwt || '', | ||||||
|  |         refreshJwt: this.agent.session.refreshJwt || '', | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       return session; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // If no agent.session but we have stored session info, return that | ||||||
|  |     if ((this as any)._sessionInfo) { | ||||||
|  |       const session = { | ||||||
|  |         did: (this as any)._sessionInfo.did, | ||||||
|  |         handle: (this as any)._sessionInfo.handle, | ||||||
|  |         accessJwt: 'dpop-protected',  // Indicate that tokens are handled by dpopFetch | ||||||
|  |         refreshJwt: 'dpop-protected', | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       return session; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |  | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   isAuthenticated(): boolean { | ||||||
|  |     return !!this.agent || !!(this as any)._sessionInfo; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getUser(): { did: string; handle: string } | null { | ||||||
|  |     const session = this.getSession(); | ||||||
|  |     if (!session) return null; | ||||||
|  |      | ||||||
|  |     return { | ||||||
|  |       did: session.did, | ||||||
|  |       handle: session.handle | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async logout(): Promise<void> { | ||||||
|  |     try { | ||||||
|  |       // Clear Agent | ||||||
|  |       this.agent = null; | ||||||
|  |        | ||||||
|  |       // Clear BrowserOAuthClient session | ||||||
|  |       if (this.oauthClient) { | ||||||
|  |         try { | ||||||
|  |           // BrowserOAuthClient may have a revoke or signOut method | ||||||
|  |           if (typeof (this.oauthClient as any).signOut === 'function') { | ||||||
|  |             await (this.oauthClient as any).signOut(); | ||||||
|  |           } else if (typeof (this.oauthClient as any).revoke === 'function') { | ||||||
|  |             await (this.oauthClient as any).revoke(); | ||||||
|  |           } | ||||||
|  |         } catch (oauthError) { | ||||||
|  |           // Ignore logout errors | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // Reset the OAuth client to force re-initialization | ||||||
|  |         this.oauthClient = null; | ||||||
|  |         this.initializePromise = null; | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       // Clear any stored session data | ||||||
|  |       localStorage.removeItem('atproto_session'); | ||||||
|  |       sessionStorage.clear(); | ||||||
|  |        | ||||||
|  |       // Clear all OAuth-related storage | ||||||
|  |       for (let i = 0; i < localStorage.length; i++) { | ||||||
|  |         const key = localStorage.key(i); | ||||||
|  |         if (key && (key.includes('oauth') || key.includes('atproto') || key.includes('session'))) { | ||||||
|  |           localStorage.removeItem(key); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       // Clear internal session info | ||||||
|  |       (this as any)._sessionInfo = null; | ||||||
|  |        | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // Force page reload to ensure clean state | ||||||
|  |       setTimeout(() => { | ||||||
|  |         window.location.reload(); | ||||||
|  |       }, 100); | ||||||
|  |        | ||||||
|  |     } catch (error) { | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // カードデータをatproto collectionに保存 | ||||||
|  |   async saveCardToBox(userCards: any[]): Promise<void> { | ||||||
|  |     // Ensure we have a valid session | ||||||
|  |     const sessionInfo = await this.checkSession(); | ||||||
|  |     if (!sessionInfo) { | ||||||
|  |       throw new Error('認証が必要です。ログインしてください。'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const did = sessionInfo.did; | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // Ensure we have a fresh agent | ||||||
|  |       if (!this.agent) { | ||||||
|  |         throw new Error('Agentが初期化されていません。'); | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       const collection = 'ai.card.box'; | ||||||
|  |       const rkey = 'self'; | ||||||
|  |       const createdAt = new Date().toISOString(); | ||||||
|  |  | ||||||
|  |       // カードボックスのレコード | ||||||
|  |       const record = { | ||||||
|  |         $type: 'ai.card.box', | ||||||
|  |         cards: userCards.map(card => ({ | ||||||
|  |           id: card.id, | ||||||
|  |           cp: card.cp, | ||||||
|  |           status: card.status, | ||||||
|  |           skill: card.skill, | ||||||
|  |           owner_did: card.owner_did, | ||||||
|  |           obtained_at: card.obtained_at, | ||||||
|  |           is_unique: card.is_unique, | ||||||
|  |           unique_id: card.unique_id | ||||||
|  |  | ||||||
|  |         })), | ||||||
|  |         total_cards: userCards.length, | ||||||
|  |         updated_at: createdAt, | ||||||
|  |         createdAt: createdAt | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |       // Use Agent's com.atproto.repo.putRecord method | ||||||
|  |       const response = await this.agent.com.atproto.repo.putRecord({ | ||||||
|  |         repo: did, | ||||||
|  |         collection: collection, | ||||||
|  |         rkey: rkey, | ||||||
|  |         record: record | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     } catch (error) { | ||||||
|  |  | ||||||
|  |       throw error; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // ai.card.boxからカード一覧を取得 | ||||||
|  |   async getCardsFromBox(): Promise<any> { | ||||||
|  |     // Ensure we have a valid session | ||||||
|  |     const sessionInfo = await this.checkSession(); | ||||||
|  |     if (!sessionInfo) { | ||||||
|  |       throw new Error('認証が必要です。ログインしてください。'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const did = sessionInfo.did; | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // Ensure we have a fresh agent | ||||||
|  |       if (!this.agent) { | ||||||
|  |         throw new Error('Agentが初期化されていません。'); | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       const response = await this.agent.com.atproto.repo.getRecord({ | ||||||
|  |         repo: did, | ||||||
|  |         collection: 'ai.card.box', | ||||||
|  |         rkey: 'self' | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // Convert to expected format | ||||||
|  |       const result = { | ||||||
|  |         records: [{ | ||||||
|  |           uri: `at://${did}/ai.card.box/self`, | ||||||
|  |           cid: response.data.cid, | ||||||
|  |           value: response.data.value | ||||||
|  |         }] | ||||||
|  |       }; | ||||||
|  |        | ||||||
|  |       return result; | ||||||
|  |     } catch (error) { | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // If record doesn't exist, return empty | ||||||
|  |       if (error.toString().includes('RecordNotFound')) { | ||||||
|  |         return { records: [] }; | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       throw error; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // ai.card.boxのコレクションを削除 | ||||||
|  |   async deleteCardBox(): Promise<void> { | ||||||
|  |     // Ensure we have a valid session | ||||||
|  |     const sessionInfo = await this.checkSession(); | ||||||
|  |     if (!sessionInfo) { | ||||||
|  |       throw new Error('認証が必要です。ログインしてください。'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const did = sessionInfo.did; | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |        | ||||||
|  |       // Ensure we have a fresh agent | ||||||
|  |       if (!this.agent) { | ||||||
|  |         throw new Error('Agentが初期化されていません。'); | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       const response = await this.agent.com.atproto.repo.deleteRecord({ | ||||||
|  |         repo: did, | ||||||
|  |         collection: 'ai.card.box', | ||||||
|  |         rkey: 'self' | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     } catch (error) { | ||||||
|  |  | ||||||
|  |       throw error; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 手動でトークンを設定(開発・デバッグ用) | ||||||
|  |   setManualTokens(accessJwt: string, refreshJwt: string): void { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |      | ||||||
|  |     // For backward compatibility, store in localStorage | ||||||
|  |     const adminDid = import.meta.env.VITE_ADMIN_DID || 'did:plc:unknown'; | ||||||
|  |     const appHost = import.meta.env.VITE_APP_HOST || 'https://example.com'; | ||||||
|  |     const session: AtprotoSession = { | ||||||
|  |       did: adminDid, | ||||||
|  |       handle: new URL(appHost).hostname, | ||||||
|  |       accessJwt: accessJwt, | ||||||
|  |       refreshJwt: refreshJwt | ||||||
|  |     }; | ||||||
|  |      | ||||||
|  |     localStorage.setItem('atproto_session', JSON.stringify(session)); | ||||||
|  |  | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 後方互換性のための従来関数 | ||||||
|  |   saveSessionToStorage(session: AtprotoSession): void { | ||||||
|  |  | ||||||
|  |     localStorage.setItem('atproto_session', JSON.stringify(session)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async backupUserCards(userCards: any[]): Promise<void> { | ||||||
|  |     return this.saveCardToBox(userCards); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const atprotoOAuthService = new AtprotoOAuthService(); | ||||||
|  | export type { AtprotoSession }; | ||||||
							
								
								
									
										109
									
								
								oauth/src/services/auth.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										109
									
								
								oauth/src/services/auth.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,109 @@ | |||||||
|  | import axios from 'axios'; | ||||||
|  |  | ||||||
|  | const API_BASE = '/api/v1'; | ||||||
|  |  | ||||||
|  | interface LoginRequest { | ||||||
|  |   identifier: string;  // Handle or DID | ||||||
|  |   password: string;    // App password | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface LoginResponse { | ||||||
|  |   access_token: string; | ||||||
|  |   token_type: string; | ||||||
|  |   did: string; | ||||||
|  |   handle: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface User { | ||||||
|  |   did: string; | ||||||
|  |   handle: string; | ||||||
|  |   avatar?: string; | ||||||
|  |   displayName?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AuthService { | ||||||
|  |   private token: string | null = null; | ||||||
|  |   private user: User | null = null; | ||||||
|  |  | ||||||
|  |   constructor() { | ||||||
|  |     // Load token from localStorage | ||||||
|  |     this.token = localStorage.getItem('ai_card_token'); | ||||||
|  |      | ||||||
|  |     // Set default auth header if token exists | ||||||
|  |     if (this.token) { | ||||||
|  |       axios.defaults.headers.common['Authorization'] = `Bearer ${this.token}`; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async login(identifier: string, password: string): Promise<LoginResponse> { | ||||||
|  |     try { | ||||||
|  |       const response = await axios.post<LoginResponse>(`${API_BASE}/auth/login`, { | ||||||
|  |         identifier, | ||||||
|  |         password | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const { access_token, did, handle } = response.data; | ||||||
|  |  | ||||||
|  |       // Store token | ||||||
|  |       this.token = access_token; | ||||||
|  |       localStorage.setItem('ai_card_token', access_token); | ||||||
|  |        | ||||||
|  |       // Set auth header | ||||||
|  |       axios.defaults.headers.common['Authorization'] = `Bearer ${access_token}`; | ||||||
|  |  | ||||||
|  |       // Store user info | ||||||
|  |       this.user = { did, handle }; | ||||||
|  |  | ||||||
|  |       return response.data; | ||||||
|  |     } catch (error) { | ||||||
|  |       throw new Error('Login failed'); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async logout(): Promise<void> { | ||||||
|  |     try { | ||||||
|  |       await axios.post(`${API_BASE}/auth/logout`); | ||||||
|  |     } catch (error) { | ||||||
|  |       // Ignore errors | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Clear token | ||||||
|  |     this.token = null; | ||||||
|  |     this.user = null; | ||||||
|  |     localStorage.removeItem('ai_card_token'); | ||||||
|  |     delete axios.defaults.headers.common['Authorization']; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async verify(): Promise<User | null> { | ||||||
|  |     if (!this.token) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       const response = await axios.get<User & { valid: boolean }>(`${API_BASE}/auth/verify`); | ||||||
|  |       if (response.data.valid) { | ||||||
|  |         this.user = { | ||||||
|  |           did: response.data.did, | ||||||
|  |           handle: response.data.handle | ||||||
|  |         }; | ||||||
|  |         return this.user; | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       // Token is invalid | ||||||
|  |       this.logout(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   getUser(): User | null { | ||||||
|  |     return this.user; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   isAuthenticated(): boolean { | ||||||
|  |     return this.token !== null; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const authService = new AuthService(); | ||||||
|  | export type { User, LoginRequest, LoginResponse }; | ||||||
							
								
								
									
										331
									
								
								oauth/src/styles/Card.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										331
									
								
								oauth/src/styles/Card.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,331 @@ | |||||||
|  | .card { | ||||||
|  |   width: 250px; | ||||||
|  |   height: 380px; | ||||||
|  |   border-radius: 12px; | ||||||
|  |   background: linear-gradient(135deg, #1a1a1a 0%, #2d2d2d 100%); | ||||||
|  |   border: 2px solid #333; | ||||||
|  |   overflow: hidden; | ||||||
|  |   position: relative; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: transform 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card:hover { | ||||||
|  |   transform: translateY(-5px); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-inner { | ||||||
|  |   padding: 20px; | ||||||
|  |   height: 100%; | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   position: relative; | ||||||
|  |   z-index: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Rarity effects */ | ||||||
|  | .card-normal { | ||||||
|  |   border-color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-rare { | ||||||
|  |   border-color: #4a90e2; | ||||||
|  |   background: linear-gradient(135deg, #1a1a2e 0%, #16213e 100%); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-super-rare { | ||||||
|  |   border-color: #9c27b0; | ||||||
|  |   background: linear-gradient(135deg, #2d1b69 0%, #0f0c29 100%); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-kira { | ||||||
|  |   border-color: #ffd700; | ||||||
|  |   background: linear-gradient(135deg, #232526 0%, #414345 100%); | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-kira::before { | ||||||
|  |   content: ''; | ||||||
|  |   position: absolute; | ||||||
|  |   top: -50%; | ||||||
|  |   left: -50%; | ||||||
|  |   width: 200%; | ||||||
|  |   height: 200%; | ||||||
|  |   background: linear-gradient( | ||||||
|  |     45deg, | ||||||
|  |     transparent 30%, | ||||||
|  |     rgba(255, 215, 0, 0.1) 50%, | ||||||
|  |     transparent 70% | ||||||
|  |   ); | ||||||
|  |   animation: shimmer 3s infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-unique { | ||||||
|  |   border-color: #ff00ff; | ||||||
|  |   background: linear-gradient(135deg, #000000 0%, #1a0033 100%); | ||||||
|  |   box-shadow: 0 0 30px rgba(255, 0, 255, 0.5); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-unique::before { | ||||||
|  |   content: ''; | ||||||
|  |   position: absolute; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   background: radial-gradient( | ||||||
|  |     circle at center, | ||||||
|  |     transparent 0%, | ||||||
|  |     rgba(255, 0, 255, 0.2) 100% | ||||||
|  |   ); | ||||||
|  |   animation: pulse 2s infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Card content */ | ||||||
|  | .card-header { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   font-size: 14px; | ||||||
|  |   color: #888; | ||||||
|  |   margin-bottom: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-image-container { | ||||||
|  |   width: 100%; | ||||||
|  |   height: 150px; | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   margin-bottom: 15px; | ||||||
|  |   overflow: hidden; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   background: rgba(255, 255, 255, 0.05); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-image { | ||||||
|  |   max-width: 100%; | ||||||
|  |   max-height: 100%; | ||||||
|  |   object-fit: contain; | ||||||
|  |   border-radius: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-content { | ||||||
|  |   flex: 1; | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-name { | ||||||
|  |   font-size: 28px; | ||||||
|  |   margin: 0; | ||||||
|  |   color: var(--card-color, #fff); | ||||||
|  |   text-align: center; | ||||||
|  |   font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .unique-badge { | ||||||
|  |   margin-top: 10px; | ||||||
|  |   padding: 5px 15px; | ||||||
|  |   background: linear-gradient(90deg, #ff00ff, #00ffff); | ||||||
|  |   border-radius: 20px; | ||||||
|  |   font-size: 12px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   animation: glow 2s ease-in-out infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-skill { | ||||||
|  |   margin-top: 20px; | ||||||
|  |   padding: 10px; | ||||||
|  |   background: rgba(255, 255, 255, 0.1); | ||||||
|  |   border-radius: 8px; | ||||||
|  |   font-size: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-footer { | ||||||
|  |   text-align: center; | ||||||
|  |   font-size: 12px; | ||||||
|  |   color: #666; | ||||||
|  |   text-transform: uppercase; | ||||||
|  |   letter-spacing: 1px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Animations */ | ||||||
|  | @keyframes shimmer { | ||||||
|  |   0% { transform: translateX(-100%); } | ||||||
|  |   100% { transform: translateX(100%); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes pulse { | ||||||
|  |   0% { opacity: 0.5; } | ||||||
|  |   50% { opacity: 1; } | ||||||
|  |   100% { opacity: 0.5; } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes glow { | ||||||
|  |   0% { box-shadow: 0 0 5px rgba(255, 0, 255, 0.5); } | ||||||
|  |   50% { box-shadow: 0 0 20px rgba(255, 0, 255, 0.8); } | ||||||
|  |   100% { box-shadow: 0 0 5px rgba(255, 0, 255, 0.5); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Simple Card Styles */ | ||||||
|  | .card-simple { | ||||||
|  |   width: 240px; | ||||||
|  |   height: auto; | ||||||
|  |   background: transparent; | ||||||
|  |   border: none; | ||||||
|  |   padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-frame { | ||||||
|  |   position: relative; | ||||||
|  |   width: 100%; | ||||||
|  |   aspect-ratio: 3/4; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   overflow: hidden; | ||||||
|  |   background: #1a1a1a; | ||||||
|  |   padding: 25px 25px 30px 25px; | ||||||
|  |   border: 3px solid #666; | ||||||
|  |   box-sizing: border-box; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Normal card - no effects */ | ||||||
|  | .card-simple.card-normal .card-frame { | ||||||
|  |   border-color: #666; | ||||||
|  |   background: #1a1a1a; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Unique (rare) card - glowing effects */ | ||||||
|  | .card-simple.card-unique .card-frame { | ||||||
|  |   border-color: #ffd700; | ||||||
|  |   background: linear-gradient(135deg, #2a2a1a 0%, #3a3a2a 50%, #2a2a1a 100%); | ||||||
|  |   position: relative; | ||||||
|  |   isolation: isolate; | ||||||
|  |   overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Particle/grainy texture for rare cards */ | ||||||
|  | .card-simple.card-unique .card-frame::before { | ||||||
|  |   content: ""; | ||||||
|  |   position: absolute; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   background-image:  | ||||||
|  |     repeating-radial-gradient(circle at 1px 1px, rgba(255, 255, 255, 0.1) 0px, transparent 1px, transparent 2px), | ||||||
|  |     repeating-radial-gradient(circle at 3px 3px, rgba(255, 215, 0, 0.1) 0px, transparent 2px, transparent 4px); | ||||||
|  |   background-size: 20px 20px, 30px 30px; | ||||||
|  |   opacity: 0.8; | ||||||
|  |   z-index: 1; | ||||||
|  |   pointer-events: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Reflection effect for rare cards */ | ||||||
|  | .card-simple.card-unique .card-frame::after { | ||||||
|  |   content: ""; | ||||||
|  |   height: 100%; | ||||||
|  |   width: 40px; | ||||||
|  |   position: absolute; | ||||||
|  |   top: -180px; | ||||||
|  |   left: 0; | ||||||
|  |   background: linear-gradient(90deg,  | ||||||
|  |     transparent 0%, | ||||||
|  |     rgba(255, 215, 0, 0.8) 20%, | ||||||
|  |     rgba(255, 255, 0, 0.9) 40%, | ||||||
|  |     rgba(255, 223, 0, 1) 50%, | ||||||
|  |     rgba(255, 255, 0, 0.9) 60%, | ||||||
|  |     rgba(255, 215, 0, 0.8) 80%, | ||||||
|  |     transparent 100% | ||||||
|  |   ); | ||||||
|  |   opacity: 0; | ||||||
|  |   transform: rotate(45deg); | ||||||
|  |   animation: gold-reflection 6s ease-in-out infinite; | ||||||
|  |   z-index: 2; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes gold-reflection { | ||||||
|  |   0% { transform: scale(0) rotate(45deg); opacity: 0; } | ||||||
|  |   15% { transform: scale(0) rotate(45deg); opacity: 0; } | ||||||
|  |   17% { transform: scale(4) rotate(45deg); opacity: 0.8; } | ||||||
|  |   20% { transform: scale(50) rotate(45deg); opacity: 0; } | ||||||
|  |   100% { transform: scale(50) rotate(45deg); opacity: 0; } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Glowing backlight effect */ | ||||||
|  | .card-simple.card-unique { | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-simple.card-unique::after { | ||||||
|  |   position: absolute; | ||||||
|  |   content: ""; | ||||||
|  |   top: 5px; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   z-index: -1; | ||||||
|  |   height: 100%; | ||||||
|  |   width: 100%; | ||||||
|  |   margin: 0 auto; | ||||||
|  |   transform: scale(0.95); | ||||||
|  |   filter: blur(15px); | ||||||
|  |   background: radial-gradient(ellipse at center, #ffd700 0%, #ffb347 50%, transparent 70%); | ||||||
|  |   opacity: 0.6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Glowing border effect for rare cards */ | ||||||
|  | .card-simple.card-unique .card-frame { | ||||||
|  |   box-shadow:  | ||||||
|  |     0 0 10px rgba(255, 215, 0, 0.5), | ||||||
|  |     inset 0 0 10px rgba(255, 215, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .card-image-simple { | ||||||
|  |   width: 100%; | ||||||
|  |   height: 100%; | ||||||
|  |   object-fit: cover; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   position: relative; | ||||||
|  |   z-index: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-cp-bar { | ||||||
|  |   width: 100%; | ||||||
|  |   height: 50px; | ||||||
|  |   background: #333; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   margin-top: 12px; | ||||||
|  |   margin-bottom: 8px; | ||||||
|  |   border: 2px solid #666; | ||||||
|  |   position: relative; | ||||||
|  |   box-sizing: border-box; | ||||||
|  |   overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-simple.card-unique .card-cp-bar { | ||||||
|  |   background: linear-gradient(135deg, #2a2a1a 0%, #3a3a2a 50%, #2a2a1a 100%); | ||||||
|  |   border-color: #ffd700; | ||||||
|  |   box-shadow:  | ||||||
|  |     0 0 5px rgba(255, 215, 0, 0.3), | ||||||
|  |     inset 0 0 5px rgba(255, 215, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .cp-value { | ||||||
|  |   font-size: 20px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   color: #fff; | ||||||
|  |   text-shadow: 0 0 10px rgba(0, 0, 0, 0.5); | ||||||
|  |   z-index: 1; | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
							
								
								
									
										196
									
								
								oauth/src/styles/CardBox.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								oauth/src/styles/CardBox.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,196 @@ | |||||||
|  | .card-box-container { | ||||||
|  |   max-width: 1200px; | ||||||
|  |   margin: 0 auto; | ||||||
|  |   padding: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-box-header { | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   align-items: center; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  |   padding-bottom: 15px; | ||||||
|  |   border-bottom: 2px solid #e9ecef; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-box-header h3 { | ||||||
|  |   color: #495057; | ||||||
|  |   margin: 0; | ||||||
|  |   font-size: 24px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .box-actions { | ||||||
|  |   display: flex; | ||||||
|  |   gap: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .uri-display { | ||||||
|  |   background: #e3f2fd; | ||||||
|  |   border: 1px solid #bbdefb; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 12px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .uri-display p { | ||||||
|  |   margin: 0; | ||||||
|  |   color: #1565c0; | ||||||
|  |   font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .uri-display code { | ||||||
|  |   background: #ffffff; | ||||||
|  |   border: 1px solid #90caf9; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   padding: 4px 8px; | ||||||
|  |   font-family: 'Monaco', 'Menlo', 'Ubuntu Mono', monospace; | ||||||
|  |   font-size: 12px; | ||||||
|  |   color: #0d47a1; | ||||||
|  |   word-break: break-all; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-button, | ||||||
|  | .refresh-button, | ||||||
|  | .retry-button, | ||||||
|  | .delete-button { | ||||||
|  |   padding: 8px 16px; | ||||||
|  |   border: none; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   font-size: 14px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-button { | ||||||
|  |   background: linear-gradient(135deg, #6f42c1 0%, #8b5fc3 100%); | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(111, 66, 193, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .refresh-button { | ||||||
|  |   background: linear-gradient(135deg, #17a2b8 0%, #20c997 100%); | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .refresh-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(23, 162, 184, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .retry-button { | ||||||
|  |   background: linear-gradient(135deg, #fd7e14 0%, #ffc107 100%); | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .retry-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(253, 126, 20, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-button { | ||||||
|  |   background: linear-gradient(135deg, #dc3545 0%, #c82333 100%); | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(220, 53, 69, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-button:disabled { | ||||||
|  |   opacity: 0.6; | ||||||
|  |   cursor: not-allowed; | ||||||
|  |   transform: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-display { | ||||||
|  |   background: #f8f9fa; | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 20px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-display h4 { | ||||||
|  |   color: #495057; | ||||||
|  |   margin-top: 0; | ||||||
|  |   margin-bottom: 15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .json-content { | ||||||
|  |   background: #ffffff; | ||||||
|  |   border: 1px solid #e9ecef; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   padding: 15px; | ||||||
|  |   font-family: 'Monaco', 'Menlo', 'Ubuntu Mono', monospace; | ||||||
|  |   font-size: 12px; | ||||||
|  |   color: #495057; | ||||||
|  |   max-height: 400px; | ||||||
|  |   overflow-y: auto; | ||||||
|  |   white-space: pre-wrap; | ||||||
|  |   word-wrap: break-word; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .box-stats { | ||||||
|  |   background: rgba(102, 126, 234, 0.1); | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 15px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .box-stats p { | ||||||
|  |   margin: 0; | ||||||
|  |   color: #495057; | ||||||
|  |   font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-grid { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); | ||||||
|  |   gap: 20px; | ||||||
|  |   margin-top: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .box-card-item { | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-info { | ||||||
|  |   margin-top: 8px; | ||||||
|  |   color: #6c757d; | ||||||
|  |   font-size: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .empty-box { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px 20px; | ||||||
|  |   color: #6c757d; | ||||||
|  |   background: #f8f9fa; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   border: 1px solid #dee2e6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .empty-box p { | ||||||
|  |   margin: 8px 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .loading, | ||||||
|  | .error { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px 20px; | ||||||
|  |   color: #6c757d; | ||||||
|  |   font-size: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .error { | ||||||
|  |   color: #dc3545; | ||||||
|  |   background: #f8d7da; | ||||||
|  |   border: 1px solid #f5c6cb; | ||||||
|  |   border-radius: 8px; | ||||||
|  | } | ||||||
							
								
								
									
										170
									
								
								oauth/src/styles/CardList.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										170
									
								
								oauth/src/styles/CardList.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,170 @@ | |||||||
|  | .card-list-container { | ||||||
|  |   min-height: 100vh; | ||||||
|  |   background: linear-gradient(135deg, #0f0f0f 0%, #1a1a1a 100%); | ||||||
|  |   padding: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-list-header { | ||||||
|  |   text-align: center; | ||||||
|  |   margin-bottom: 40px; | ||||||
|  |   padding: 20px; | ||||||
|  |   background: rgba(255, 255, 255, 0.05); | ||||||
|  |   border-radius: 12px; | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-list-header h1 { | ||||||
|  |   color: #fff; | ||||||
|  |   margin: 0 0 10px 0; | ||||||
|  |   font-size: 2.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-list-header p { | ||||||
|  |   color: #999; | ||||||
|  |   margin: 0; | ||||||
|  |   font-size: 1.1rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-list-grid { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: repeat(auto-fill, minmax(300px, 1fr)); | ||||||
|  |   gap: 30px; | ||||||
|  |   max-width: 1400px; | ||||||
|  |   margin: 0 auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-list-item { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Simple grid layout for user-page style */ | ||||||
|  | .card-list-simple-grid { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); | ||||||
|  |   gap: 20px; | ||||||
|  |   max-width: 1400px; | ||||||
|  |   margin: 0 auto; | ||||||
|  |   padding: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-list-simple-item { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .info-button { | ||||||
|  |   background: linear-gradient(135deg, #333 0%, #555 100%); | ||||||
|  |   color: white; | ||||||
|  |   border: 2px solid #666; | ||||||
|  |   padding: 8px 16px; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 0.9rem; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   width: 100%; | ||||||
|  |   max-width: 240px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .info-button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3); | ||||||
|  |   background: linear-gradient(135deg, #444 0%, #666 100%); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-info-details { | ||||||
|  |   background: rgba(255, 255, 255, 0.05); | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.1); | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 15px; | ||||||
|  |   width: 100%; | ||||||
|  |   max-width: 240px; | ||||||
|  |   margin-top: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-info-details p { | ||||||
|  |   margin: 5px 0; | ||||||
|  |   color: #ccc; | ||||||
|  |   font-size: 0.85rem; | ||||||
|  |   text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-info-details p strong { | ||||||
|  |   color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-meta { | ||||||
|  |   background: rgba(255, 255, 255, 0.05); | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.1); | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 15px; | ||||||
|  |   width: 100%; | ||||||
|  |   max-width: 250px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-meta p { | ||||||
|  |   margin: 5px 0; | ||||||
|  |   color: #ccc; | ||||||
|  |   font-size: 0.9rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-meta p:first-child { | ||||||
|  |   font-weight: bold; | ||||||
|  |   color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-description { | ||||||
|  |   font-size: 0.85rem; | ||||||
|  |   color: #999; | ||||||
|  |   font-style: italic; | ||||||
|  |   margin-top: 8px; | ||||||
|  |   line-height: 1.4; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .source-info { | ||||||
|  |   font-size: 0.9rem; | ||||||
|  |   color: #666; | ||||||
|  |   margin-top: 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .loading, .error { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px; | ||||||
|  |   color: #999; | ||||||
|  |   font-size: 1.2rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .error { | ||||||
|  |   color: #ff4757; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | button { | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   padding: 10px 20px; | ||||||
|  |   border-radius: 5px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   font-size: 1rem; | ||||||
|  |   margin-top: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | button:hover { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 4px 12px rgba(102, 126, 234, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media (max-width: 768px) { | ||||||
|  |   .card-list-grid { | ||||||
|  |     grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); | ||||||
|  |     gap: 20px; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   .card-list-header h1 { | ||||||
|  |     font-size: 2rem; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										172
									
								
								oauth/src/styles/CollectionAnalysis.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								oauth/src/styles/CollectionAnalysis.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,172 @@ | |||||||
|  | .collection-analysis { | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   border-radius: 16px; | ||||||
|  |   padding: 24px; | ||||||
|  |   margin: 20px 0; | ||||||
|  |   color: white; | ||||||
|  |   box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .collection-analysis h3 { | ||||||
|  |   margin: 0 0 20px 0; | ||||||
|  |   font-size: 1.5rem; | ||||||
|  |   font-weight: 600; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .analysis-stats { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: repeat(auto-fit, minmax(120px, 1fr)); | ||||||
|  |   gap: 16px; | ||||||
|  |   margin-bottom: 24px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stat-card { | ||||||
|  |   background: rgba(255, 255, 255, 0.15); | ||||||
|  |   backdrop-filter: blur(10px); | ||||||
|  |   border-radius: 12px; | ||||||
|  |   padding: 16px; | ||||||
|  |   text-align: center; | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stat-value { | ||||||
|  |   font-size: 2rem; | ||||||
|  |   font-weight: bold; | ||||||
|  |   margin-bottom: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stat-label { | ||||||
|  |   font-size: 0.9rem; | ||||||
|  |   opacity: 0.8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-distribution { | ||||||
|  |   margin-bottom: 24px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-distribution h4 { | ||||||
|  |   margin: 0 0 16px 0; | ||||||
|  |   font-size: 1.2rem; | ||||||
|  |   font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-bars { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   gap: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-bar { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-name { | ||||||
|  |   min-width: 80px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   text-transform: capitalize; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .bar-container { | ||||||
|  |   flex: 1; | ||||||
|  |   height: 20px; | ||||||
|  |   background: rgba(255, 255, 255, 0.2); | ||||||
|  |   border-radius: 10px; | ||||||
|  |   overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .bar { | ||||||
|  |   height: 100%; | ||||||
|  |   border-radius: 10px; | ||||||
|  |   transition: width 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .bar-common { background: linear-gradient(90deg, #4CAF50, #45a049); } | ||||||
|  | .bar-rare { background: linear-gradient(90deg, #2196F3, #1976D2); } | ||||||
|  | .bar-epic { background: linear-gradient(90deg, #9C27B0, #7B1FA2); } | ||||||
|  | .bar-legendary { background: linear-gradient(90deg, #FF9800, #F57C00); } | ||||||
|  | .bar-mythic { background: linear-gradient(90deg, #F44336, #D32F2F); } | ||||||
|  |  | ||||||
|  | .rarity-count { | ||||||
|  |   min-width: 40px; | ||||||
|  |   text-align: right; | ||||||
|  |   font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .recommendations { | ||||||
|  |   background: rgba(255, 255, 255, 0.1); | ||||||
|  |   border-radius: 12px; | ||||||
|  |   padding: 16px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .recommendations h4 { | ||||||
|  |   margin: 0 0 12px 0; | ||||||
|  |   font-size: 1.1rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .recommendations ul { | ||||||
|  |   margin: 0; | ||||||
|  |   padding-left: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .recommendations li { | ||||||
|  |   margin-bottom: 8px; | ||||||
|  |   line-height: 1.4; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .refresh-analysis, | ||||||
|  | .analyze-button, | ||||||
|  | .retry-button { | ||||||
|  |   background: rgba(255, 255, 255, 0.2); | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.3); | ||||||
|  |   color: white; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 12px 24px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   display: block; | ||||||
|  |   margin: 0 auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .refresh-analysis:hover, | ||||||
|  | .analyze-button:hover, | ||||||
|  | .retry-button:hover { | ||||||
|  |   background: rgba(255, 255, 255, 0.3); | ||||||
|  |   transform: translateY(-2px); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .analysis-loading, | ||||||
|  | .analysis-error, | ||||||
|  | .analysis-empty { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .loading-spinner { | ||||||
|  |   width: 40px; | ||||||
|  |   height: 40px; | ||||||
|  |   border: 3px solid rgba(255, 255, 255, 0.3); | ||||||
|  |   border-top: 3px solid white; | ||||||
|  |   border-radius: 50%; | ||||||
|  |   animation: spin 1s linear infinite; | ||||||
|  |   margin: 0 auto 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes spin { | ||||||
|  |   0% { transform: rotate(0deg); } | ||||||
|  |   100% { transform: rotate(360deg); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .analysis-error p { | ||||||
|  |   color: #ffcdd2; | ||||||
|  |   margin-bottom: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .analysis-empty p { | ||||||
|  |   opacity: 0.8; | ||||||
|  |   margin-bottom: 16px; | ||||||
|  | } | ||||||
							
								
								
									
										174
									
								
								oauth/src/styles/GachaAnimation.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								oauth/src/styles/GachaAnimation.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | |||||||
|  | .gacha-container { | ||||||
|  |   position: fixed; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   background: rgba(0, 0, 0, 0.9); | ||||||
|  |   z-index: 1000; | ||||||
|  |   cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-final { | ||||||
|  |   position: relative; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-actions { | ||||||
|  |   position: absolute; | ||||||
|  |   bottom: -80px; | ||||||
|  |   left: 50%; | ||||||
|  |   transform: translateX(-50%); | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .save-button { | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   color: white; | ||||||
|  |   border: none; | ||||||
|  |   padding: 10px 20px; | ||||||
|  |   border-radius: 25px; | ||||||
|  |   font-size: 14px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   box-shadow: 0 4px 15px rgba(102, 126, 234, 0.3); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .save-button:hover:not(:disabled) { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 6px 20px rgba(102, 126, 234, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .save-button:disabled { | ||||||
|  |   opacity: 0.6; | ||||||
|  |   cursor: not-allowed; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .click-hint { | ||||||
|  |   color: white; | ||||||
|  |   font-size: 12px; | ||||||
|  |   background: rgba(0, 0, 0, 0.7); | ||||||
|  |   padding: 6px 12px; | ||||||
|  |   border-radius: 15px; | ||||||
|  |   animation: pulse 2s infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes pulse { | ||||||
|  |   0%, 100% { opacity: 0.7; } | ||||||
|  |   50% { opacity: 1; } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-opening { | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-pack { | ||||||
|  |   width: 200px; | ||||||
|  |   height: 280px; | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   border-radius: 16px; | ||||||
|  |   position: relative; | ||||||
|  |   box-shadow: 0 10px 40px rgba(0, 0, 0, 0.5); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .pack-glow { | ||||||
|  |   position: absolute; | ||||||
|  |   top: -20px; | ||||||
|  |   left: -20px; | ||||||
|  |   right: -20px; | ||||||
|  |   bottom: -20px; | ||||||
|  |   background: radial-gradient(circle, rgba(255, 255, 255, 0.3) 0%, transparent 70%); | ||||||
|  |   animation: glow-pulse 2s ease-in-out infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Effect variations */ | ||||||
|  | .effect-normal { | ||||||
|  |   background: radial-gradient(circle, rgba(255, 255, 255, 0.1) 0%, transparent 50%); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .effect-rare { | ||||||
|  |   background: radial-gradient(circle, rgba(74, 144, 226, 0.2) 0%, transparent 50%); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .effect-kira { | ||||||
|  |   background: radial-gradient(circle, rgba(255, 215, 0, 0.3) 0%, transparent 50%); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .effect-kira::before { | ||||||
|  |   content: ''; | ||||||
|  |   position: absolute; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   background: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100"><polygon points="50,0 60,40 100,50 60,60 50,100 40,60 0,50 40,40" fill="rgba(255,215,0,0.1)"/></svg>'); | ||||||
|  |   background-size: 50px 50px; | ||||||
|  |   animation: sparkle 3s linear infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .effect-unique { | ||||||
|  |   background: radial-gradient(circle, rgba(255, 0, 255, 0.4) 0%, transparent 50%); | ||||||
|  |   overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .unique-effect { | ||||||
|  |   position: absolute; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   pointer-events: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .unique-particles { | ||||||
|  |   position: absolute; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   background-image:  | ||||||
|  |     radial-gradient(circle, #ff00ff 1px, transparent 1px), | ||||||
|  |     radial-gradient(circle, #00ffff 1px, transparent 1px); | ||||||
|  |   background-size: 50px 50px, 30px 30px; | ||||||
|  |   background-position: 0 0, 25px 25px; | ||||||
|  |   animation: particle-float 20s linear infinite; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .unique-burst { | ||||||
|  |   position: absolute; | ||||||
|  |   top: 50%; | ||||||
|  |   left: 50%; | ||||||
|  |   width: 300px; | ||||||
|  |   height: 300px; | ||||||
|  |   transform: translate(-50%, -50%); | ||||||
|  |   background: radial-gradient(circle, rgba(255, 0, 255, 0.8) 0%, transparent 70%); | ||||||
|  |   animation: burst 1s ease-out; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Animations */ | ||||||
|  | @keyframes glow-pulse { | ||||||
|  |   0%, 100% { opacity: 0.5; transform: scale(1); } | ||||||
|  |   50% { opacity: 1; transform: scale(1.1); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes sparkle { | ||||||
|  |   0% { transform: translateY(0) rotate(0deg); } | ||||||
|  |   100% { transform: translateY(-100vh) rotate(360deg); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes particle-float { | ||||||
|  |   0% { transform: translate(0, 0); } | ||||||
|  |   100% { transform: translate(-50px, -100px); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes burst { | ||||||
|  |   0% { transform: translate(-50%, -50%) scale(0); opacity: 1; } | ||||||
|  |   100% { transform: translate(-50%, -50%) scale(3); opacity: 0; } | ||||||
|  | } | ||||||
							
								
								
									
										219
									
								
								oauth/src/styles/GachaStats.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										219
									
								
								oauth/src/styles/GachaStats.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,219 @@ | |||||||
|  | .gacha-stats { | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   border-radius: 16px; | ||||||
|  |   padding: 24px; | ||||||
|  |   margin: 20px 0; | ||||||
|  |   color: white; | ||||||
|  |   box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .gacha-stats h3 { | ||||||
|  |   margin: 0 0 20px 0; | ||||||
|  |   font-size: 1.5rem; | ||||||
|  |   font-weight: 600; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stats-overview { | ||||||
|  |   margin-bottom: 24px; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .overview-card { | ||||||
|  |   background: rgba(255, 255, 255, 0.15); | ||||||
|  |   backdrop-filter: blur(10px); | ||||||
|  |   border-radius: 12px; | ||||||
|  |   padding: 20px; | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.2); | ||||||
|  |   display: inline-block; | ||||||
|  |   min-width: 200px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .overview-value { | ||||||
|  |   font-size: 2.5rem; | ||||||
|  |   font-weight: bold; | ||||||
|  |   margin-bottom: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .overview-label { | ||||||
|  |   font-size: 1rem; | ||||||
|  |   opacity: 0.9; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-stats { | ||||||
|  |   margin-bottom: 24px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-stats h4 { | ||||||
|  |   margin: 0 0 16px 0; | ||||||
|  |   font-size: 1.2rem; | ||||||
|  |   font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-grid { | ||||||
|  |   display: grid; | ||||||
|  |   grid-template-columns: repeat(auto-fit, minmax(120px, 1fr)); | ||||||
|  |   gap: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-stat { | ||||||
|  |   background: rgba(255, 255, 255, 0.15); | ||||||
|  |   backdrop-filter: blur(10px); | ||||||
|  |   border-radius: 12px; | ||||||
|  |   padding: 16px; | ||||||
|  |   text-align: center; | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.2); | ||||||
|  |   position: relative; | ||||||
|  |   overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-stat::before { | ||||||
|  |   content: ''; | ||||||
|  |   position: absolute; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   height: 3px; | ||||||
|  |   background: var(--rarity-color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-stat.rarity-common { --rarity-color: #4CAF50; } | ||||||
|  | .rarity-stat.rarity-rare { --rarity-color: #2196F3; } | ||||||
|  | .rarity-stat.rarity-epic { --rarity-color: #9C27B0; } | ||||||
|  | .rarity-stat.rarity-legendary { --rarity-color: #FF9800; } | ||||||
|  | .rarity-stat.rarity-mythic { --rarity-color: #F44336; } | ||||||
|  |  | ||||||
|  | .rarity-count { | ||||||
|  |   font-size: 1.8rem; | ||||||
|  |   font-weight: bold; | ||||||
|  |   margin-bottom: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .rarity-name { | ||||||
|  |   font-size: 0.9rem; | ||||||
|  |   opacity: 0.9; | ||||||
|  |   text-transform: capitalize; | ||||||
|  |   margin-bottom: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .success-rate { | ||||||
|  |   font-size: 0.8rem; | ||||||
|  |   opacity: 0.7; | ||||||
|  |   background: rgba(255, 255, 255, 0.1); | ||||||
|  |   border-radius: 4px; | ||||||
|  |   padding: 2px 6px; | ||||||
|  |   display: inline-block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .recent-activity { | ||||||
|  |   background: rgba(255, 255, 255, 0.1); | ||||||
|  |   border-radius: 12px; | ||||||
|  |   padding: 16px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .recent-activity h4 { | ||||||
|  |   margin: 0 0 12px 0; | ||||||
|  |   font-size: 1.1rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .activity-list { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   gap: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .activity-item { | ||||||
|  |   background: rgba(255, 255, 255, 0.05); | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 12px; | ||||||
|  |   display: flex; | ||||||
|  |   justify-content: space-between; | ||||||
|  |   align-items: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .activity-time { | ||||||
|  |   font-size: 0.8rem; | ||||||
|  |   opacity: 0.7; | ||||||
|  |   min-width: 120px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .activity-details { | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   gap: 8px; | ||||||
|  |   flex: 1; | ||||||
|  |   justify-content: flex-end; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-rarity { | ||||||
|  |   padding: 2px 8px; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   font-size: 0.75rem; | ||||||
|  |   font-weight: 500; | ||||||
|  |   text-transform: uppercase; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .card-rarity.rarity-common { background: #4CAF50; } | ||||||
|  | .card-rarity.rarity-rare { background: #2196F3; } | ||||||
|  | .card-rarity.rarity-epic { background: #9C27B0; } | ||||||
|  | .card-rarity.rarity-legendary { background: #FF9800; } | ||||||
|  | .card-rarity.rarity-mythic { background: #F44336; } | ||||||
|  |  | ||||||
|  | .card-name { | ||||||
|  |   font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .refresh-stats, | ||||||
|  | .load-stats-button, | ||||||
|  | .retry-button { | ||||||
|  |   background: rgba(255, 255, 255, 0.2); | ||||||
|  |   border: 1px solid rgba(255, 255, 255, 0.3); | ||||||
|  |   color: white; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 12px 24px; | ||||||
|  |   font-weight: 500; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   display: block; | ||||||
|  |   margin: 0 auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .refresh-stats:hover, | ||||||
|  | .load-stats-button:hover, | ||||||
|  | .retry-button:hover { | ||||||
|  |   background: rgba(255, 255, 255, 0.3); | ||||||
|  |   transform: translateY(-2px); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stats-loading, | ||||||
|  | .stats-error, | ||||||
|  | .stats-empty { | ||||||
|  |   text-align: center; | ||||||
|  |   padding: 40px 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .loading-spinner { | ||||||
|  |   width: 40px; | ||||||
|  |   height: 40px; | ||||||
|  |   border: 3px solid rgba(255, 255, 255, 0.3); | ||||||
|  |   border-top: 3px solid white; | ||||||
|  |   border-radius: 50%; | ||||||
|  |   animation: spin 1s linear infinite; | ||||||
|  |   margin: 0 auto 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @keyframes spin { | ||||||
|  |   0% { transform: rotate(0deg); } | ||||||
|  |   100% { transform: rotate(360deg); } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stats-error p { | ||||||
|  |   color: #ffcdd2; | ||||||
|  |   margin-bottom: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stats-empty p { | ||||||
|  |   opacity: 0.8; | ||||||
|  |   margin-bottom: 16px; | ||||||
|  | } | ||||||
							
								
								
									
										243
									
								
								oauth/src/styles/Login.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										243
									
								
								oauth/src/styles/Login.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,243 @@ | |||||||
|  | .login-overlay { | ||||||
|  |   position: fixed; | ||||||
|  |   top: 0; | ||||||
|  |   left: 0; | ||||||
|  |   right: 0; | ||||||
|  |   bottom: 0; | ||||||
|  |   background: rgba(0, 0, 0, 0.8); | ||||||
|  |   display: flex; | ||||||
|  |   align-items: center; | ||||||
|  |   justify-content: center; | ||||||
|  |   z-index: 1000; | ||||||
|  |   backdrop-filter: blur(5px); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-modal { | ||||||
|  |   background: linear-gradient(135deg, #1a1a1a 0%, #2d2d2d 100%); | ||||||
|  |   border: 1px solid #444; | ||||||
|  |   border-radius: 16px; | ||||||
|  |   padding: 40px; | ||||||
|  |   max-width: 450px; | ||||||
|  |   width: 90%; | ||||||
|  |   box-shadow: 0 20px 60px rgba(0, 0, 0, 0.5); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-mode-selector { | ||||||
|  |   display: flex; | ||||||
|  |   margin-bottom: 24px; | ||||||
|  |   background: rgba(255, 255, 255, 0.05); | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .mode-button { | ||||||
|  |   flex: 1; | ||||||
|  |   padding: 12px 16px; | ||||||
|  |   border: none; | ||||||
|  |   background: transparent; | ||||||
|  |   color: #ccc; | ||||||
|  |   border-radius: 6px; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .mode-button.active { | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   color: white; | ||||||
|  |   box-shadow: 0 2px 8px rgba(102, 126, 234, 0.3); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .mode-button:hover:not(.active) { | ||||||
|  |   background: rgba(255, 255, 255, 0.1); | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-login { | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-info { | ||||||
|  |   margin-bottom: 24px; | ||||||
|  |   padding: 20px; | ||||||
|  |   background: rgba(102, 126, 234, 0.1); | ||||||
|  |   border-radius: 12px; | ||||||
|  |   border: 1px solid rgba(102, 126, 234, 0.3); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-info h3 { | ||||||
|  |   margin: 0 0 12px 0; | ||||||
|  |   font-size: 18px; | ||||||
|  |   color: #667eea; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-info p { | ||||||
|  |   margin: 0; | ||||||
|  |   font-size: 14px; | ||||||
|  |   line-height: 1.5; | ||||||
|  |   opacity: 0.9; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-login-button { | ||||||
|  |   background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | ||||||
|  |   border: none; | ||||||
|  |   color: white; | ||||||
|  |   padding: 16px 32px; | ||||||
|  |   border-radius: 12px; | ||||||
|  |   font-size: 16px; | ||||||
|  |   font-weight: 600; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  |   box-shadow: 0 4px 16px rgba(102, 126, 234, 0.3); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-login-button:hover:not(:disabled) { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 6px 20px rgba(102, 126, 234, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .oauth-login-button:disabled { | ||||||
|  |   opacity: 0.7; | ||||||
|  |   cursor: not-allowed; | ||||||
|  |   transform: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-modal h2 { | ||||||
|  |   margin: 0 0 30px 0; | ||||||
|  |   font-size: 28px; | ||||||
|  |   text-align: center; | ||||||
|  |   background: linear-gradient(90deg, #fff700 0%, #ff00ff 100%); | ||||||
|  |   -webkit-background-clip: text; | ||||||
|  |   -webkit-text-fill-color: transparent; | ||||||
|  |   background-clip: text; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group { | ||||||
|  |   margin-bottom: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group label { | ||||||
|  |   display: block; | ||||||
|  |   margin-bottom: 8px; | ||||||
|  |   color: #ccc; | ||||||
|  |   font-size: 14px; | ||||||
|  |   font-weight: 500; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group input { | ||||||
|  |   width: 100%; | ||||||
|  |   padding: 12px 16px; | ||||||
|  |   background: rgba(255, 255, 255, 0.1); | ||||||
|  |   border: 1px solid #444; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   color: white; | ||||||
|  |   font-size: 16px; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group input:focus { | ||||||
|  |   outline: none; | ||||||
|  |   border-color: #fff700; | ||||||
|  |   background: rgba(255, 255, 255, 0.15); | ||||||
|  |   box-shadow: 0 0 0 2px rgba(255, 247, 0, 0.2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group input:disabled { | ||||||
|  |   opacity: 0.5; | ||||||
|  |   cursor: not-allowed; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group small { | ||||||
|  |   display: block; | ||||||
|  |   margin-top: 6px; | ||||||
|  |   color: #888; | ||||||
|  |   font-size: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group small a { | ||||||
|  |   color: #fff700; | ||||||
|  |   text-decoration: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-group small a:hover { | ||||||
|  |   text-decoration: underline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .error-message { | ||||||
|  |   background: rgba(255, 71, 87, 0.1); | ||||||
|  |   border: 1px solid rgba(255, 71, 87, 0.3); | ||||||
|  |   border-radius: 8px; | ||||||
|  |   padding: 12px; | ||||||
|  |   margin-bottom: 20px; | ||||||
|  |   color: #ff4757; | ||||||
|  |   font-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button-group { | ||||||
|  |   display: flex; | ||||||
|  |   gap: 12px; | ||||||
|  |   margin-top: 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-button, | ||||||
|  | .cancel-button { | ||||||
|  |   flex: 1; | ||||||
|  |   padding: 14px 24px; | ||||||
|  |   border: none; | ||||||
|  |   border-radius: 8px; | ||||||
|  |   font-size: 16px; | ||||||
|  |   font-weight: bold; | ||||||
|  |   cursor: pointer; | ||||||
|  |   transition: all 0.3s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-button { | ||||||
|  |   background: linear-gradient(135deg, #fff700 0%, #ffd700 100%); | ||||||
|  |   color: #000; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-button:hover:not(:disabled) { | ||||||
|  |   transform: translateY(-2px); | ||||||
|  |   box-shadow: 0 6px 20px rgba(255, 247, 0, 0.4); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-button:disabled { | ||||||
|  |   opacity: 0.5; | ||||||
|  |   cursor: not-allowed; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .cancel-button { | ||||||
|  |   background: rgba(255, 255, 255, 0.1); | ||||||
|  |   color: white; | ||||||
|  |   border: 1px solid #444; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .cancel-button:hover:not(:disabled) { | ||||||
|  |   background: rgba(255, 255, 255, 0.15); | ||||||
|  |   border-color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-info { | ||||||
|  |   margin-top: 30px; | ||||||
|  |   padding-top: 20px; | ||||||
|  |   border-top: 1px solid #333; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login-info p { | ||||||
|  |   color: #888; | ||||||
|  |   font-size: 14px; | ||||||
|  |   line-height: 1.6; | ||||||
|  |   margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .dev-notice { | ||||||
|  |   background: rgba(255, 193, 7, 0.1); | ||||||
|  |   border: 1px solid rgba(255, 193, 7, 0.3); | ||||||
|  |   border-radius: 6px; | ||||||
|  |   padding: 8px 12px; | ||||||
|  |   margin: 10px 0; | ||||||
|  |   color: #ffc107; | ||||||
|  |   font-size: 12px; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
							
								
								
									
										135
									
								
								oauth/src/tests/console-test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								oauth/src/tests/console-test.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,135 @@ | |||||||
|  | // Simple console test for OAuth app | ||||||
|  | // This runs before 'npm run preview' to display test results | ||||||
|  |  | ||||||
|  | // Mock import.meta.env for Node.js environment | ||||||
|  | (global as any).import = { | ||||||
|  |   meta: { | ||||||
|  |     env: { | ||||||
|  |       VITE_ATPROTO_PDS: process.env.VITE_ATPROTO_PDS || 'syu.is', | ||||||
|  |       VITE_ADMIN_HANDLE: process.env.VITE_ADMIN_HANDLE || 'ai.syui.ai', | ||||||
|  |       VITE_AI_HANDLE: process.env.VITE_AI_HANDLE || 'ai.syui.ai', | ||||||
|  |       VITE_OAUTH_COLLECTION: process.env.VITE_OAUTH_COLLECTION || 'ai.syui.log', | ||||||
|  |       VITE_ATPROTO_HANDLE_LIST: process.env.VITE_ATPROTO_HANDLE_LIST || '["syui.ai", "ai.syui.ai", "yui.syui.ai"]', | ||||||
|  |       VITE_APP_HOST: process.env.VITE_APP_HOST || 'https://log.syui.ai' | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Simple implementation of functions for testing | ||||||
|  | function detectPdsFromHandle(handle: string): string { | ||||||
|  |   if (handle.endsWith('.syu.is') || handle.endsWith('.syui.ai')) { | ||||||
|  |     return 'syu.is'; | ||||||
|  |   } | ||||||
|  |   if (handle.endsWith('.bsky.social')) { | ||||||
|  |     return 'bsky.social'; | ||||||
|  |   } | ||||||
|  |   // Default case - check if it's in the allowed list | ||||||
|  |   const allowedHandles = JSON.parse((global as any).import.meta.env.VITE_ATPROTO_HANDLE_LIST || '[]'); | ||||||
|  |   if (allowedHandles.includes(handle)) { | ||||||
|  |     return (global as any).import.meta.env.VITE_ATPROTO_PDS || 'syu.is'; | ||||||
|  |   } | ||||||
|  |   return 'bsky.social'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function getNetworkConfig(pds: string) { | ||||||
|  |   switch (pds) { | ||||||
|  |     case 'bsky.social': | ||||||
|  |     case 'bsky.app': | ||||||
|  |       return { | ||||||
|  |         pdsApi: `https://${pds}`, | ||||||
|  |         plcApi: 'https://plc.directory', | ||||||
|  |         bskyApi: 'https://public.api.bsky.app', | ||||||
|  |         webUrl: 'https://bsky.app' | ||||||
|  |       }; | ||||||
|  |     case 'syu.is': | ||||||
|  |       return { | ||||||
|  |         pdsApi: 'https://syu.is', | ||||||
|  |         plcApi: 'https://plc.syu.is', | ||||||
|  |         bskyApi: 'https://bsky.syu.is', | ||||||
|  |         webUrl: 'https://web.syu.is' | ||||||
|  |       }; | ||||||
|  |     default: | ||||||
|  |       return { | ||||||
|  |         pdsApi: `https://${pds}`, | ||||||
|  |         plcApi: 'https://plc.directory', | ||||||
|  |         bskyApi: 'https://public.api.bsky.app', | ||||||
|  |         webUrl: 'https://bsky.app' | ||||||
|  |       }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Main test execution | ||||||
|  | console.log('\n=== OAuth App Configuration Tests ===\n'); | ||||||
|  |  | ||||||
|  | // Test 1: Handle input behavior | ||||||
|  | console.log('1. Handle Input → PDS Detection:'); | ||||||
|  | const testHandles = [ | ||||||
|  |   'syui.ai', | ||||||
|  |   'syui.syu.is',  | ||||||
|  |   'syui.syui.ai', | ||||||
|  |   'test.bsky.social', | ||||||
|  |   'unknown.handle' | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | testHandles.forEach(handle => { | ||||||
|  |   const pds = detectPdsFromHandle(handle); | ||||||
|  |   const config = getNetworkConfig(pds); | ||||||
|  |   console.log(`   ${handle.padEnd(20)} → PDS: ${pds.padEnd(12)} → API: ${config.pdsApi}`); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Test 2: Environment variable impact | ||||||
|  | console.log('\n2. Current Environment Configuration:'); | ||||||
|  | const env = (global as any).import.meta.env; | ||||||
|  | console.log(`   VITE_ATPROTO_PDS:      ${env.VITE_ATPROTO_PDS}`); | ||||||
|  | console.log(`   VITE_ADMIN_HANDLE:     ${env.VITE_ADMIN_HANDLE}`); | ||||||
|  | console.log(`   VITE_AI_HANDLE:        ${env.VITE_AI_HANDLE}`); | ||||||
|  | console.log(`   VITE_OAUTH_COLLECTION: ${env.VITE_OAUTH_COLLECTION}`); | ||||||
|  | console.log(`   VITE_ATPROTO_HANDLE_LIST: ${env.VITE_ATPROTO_HANDLE_LIST}`); | ||||||
|  |  | ||||||
|  | // Test 3: API endpoint generation | ||||||
|  | console.log('\n3. Generated API Endpoints:'); | ||||||
|  | const adminPds = detectPdsFromHandle(env.VITE_ADMIN_HANDLE); | ||||||
|  | const adminConfig = getNetworkConfig(adminPds); | ||||||
|  | console.log(`   Admin PDS detection: ${env.VITE_ADMIN_HANDLE} → ${adminPds}`); | ||||||
|  | console.log(`   Admin API endpoints:`); | ||||||
|  | console.log(`     - PDS API:  ${adminConfig.pdsApi}`); | ||||||
|  | console.log(`     - Bsky API: ${adminConfig.bskyApi}`); | ||||||
|  | console.log(`     - Web URL:  ${adminConfig.webUrl}`); | ||||||
|  |  | ||||||
|  | // Test 4: Collection URLs | ||||||
|  | console.log('\n4. Collection API URLs:'); | ||||||
|  | const baseCollection = env.VITE_OAUTH_COLLECTION; | ||||||
|  | console.log(`   User list: ${adminConfig.pdsApi}/xrpc/com.atproto.repo.listRecords?repo=${env.VITE_ADMIN_HANDLE}&collection=${baseCollection}.user`); | ||||||
|  | console.log(`   Chat:      ${adminConfig.pdsApi}/xrpc/com.atproto.repo.listRecords?repo=${env.VITE_ADMIN_HANDLE}&collection=${baseCollection}.chat`); | ||||||
|  | console.log(`   Lang:      ${adminConfig.pdsApi}/xrpc/com.atproto.repo.listRecords?repo=${env.VITE_ADMIN_HANDLE}&collection=${baseCollection}.chat.lang`); | ||||||
|  | console.log(`   Comment:   ${adminConfig.pdsApi}/xrpc/com.atproto.repo.listRecords?repo=${env.VITE_ADMIN_HANDLE}&collection=${baseCollection}.chat.comment`); | ||||||
|  |  | ||||||
|  | // Test 5: OAuth routing logic | ||||||
|  | console.log('\n5. OAuth Authorization Logic:'); | ||||||
|  | const allowedHandles = JSON.parse(env.VITE_ATPROTO_HANDLE_LIST || '[]'); | ||||||
|  | console.log(`   Allowed handles: ${JSON.stringify(allowedHandles)}`); | ||||||
|  | console.log(`   OAuth scenarios:`); | ||||||
|  |  | ||||||
|  | const oauthTestCases = [ | ||||||
|  |   'syui.ai',         // Should use syu.is (in allowed list) | ||||||
|  |   'test.syu.is',     // Should use syu.is (*.syu.is pattern) | ||||||
|  |   'user.bsky.social' // Should use bsky.social (default) | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | oauthTestCases.forEach(handle => { | ||||||
|  |   const pds = detectPdsFromHandle(handle); | ||||||
|  |   const isAllowed = allowedHandles.includes(handle); | ||||||
|  |   const reason = handle.endsWith('.syu.is') ? '*.syu.is pattern' :  | ||||||
|  |                  isAllowed ? 'in allowed list' :  | ||||||
|  |                  'default'; | ||||||
|  |   console.log(`     ${handle.padEnd(20)} → https://${pds}/oauth/authorize (${reason})`); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Test 6: AI Profile Resolution | ||||||
|  | console.log('\n6. AI Profile Resolution:'); | ||||||
|  | const aiPds = detectPdsFromHandle(env.VITE_AI_HANDLE); | ||||||
|  | const aiConfig = getNetworkConfig(aiPds); | ||||||
|  | console.log(`   AI Handle: ${env.VITE_AI_HANDLE} → PDS: ${aiPds}`); | ||||||
|  | console.log(`   AI Profile API: ${aiConfig.bskyApi}/xrpc/app.bsky.actor.getProfile?actor=${env.VITE_AI_HANDLE}`); | ||||||
|  |  | ||||||
|  | console.log('\n=== Tests Complete ===\n'); | ||||||
							
								
								
									
										141
									
								
								oauth/src/tests/oauth.test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								oauth/src/tests/oauth.test.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | |||||||
|  | import { describe, it, expect, beforeEach } from 'vitest'; | ||||||
|  | import { getAppConfig } from '../config/app'; | ||||||
|  | import { detectPdsFromHandle, getNetworkConfig } from '../App'; | ||||||
|  |  | ||||||
|  | // Test helper to mock environment variables | ||||||
|  | const mockEnv = (vars: Record<string, string>) => { | ||||||
|  |   Object.keys(vars).forEach(key => { | ||||||
|  |     (import.meta.env as any)[key] = vars[key]; | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | describe('OAuth App Tests', () => { | ||||||
|  |   describe('Handle Input Behavior', () => { | ||||||
|  |     it('should detect PDS for syui.ai (Bluesky)', () => { | ||||||
|  |       const pds = detectPdsFromHandle('syui.ai'); | ||||||
|  |       expect(pds).toBe('bsky.social'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('should detect PDS for syui.syu.is (syu.is)', () => { | ||||||
|  |       const pds = detectPdsFromHandle('syui.syu.is'); | ||||||
|  |       expect(pds).toBe('syu.is'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('should detect PDS for syui.syui.ai (syu.is)', () => { | ||||||
|  |       const pds = detectPdsFromHandle('syui.syui.ai'); | ||||||
|  |       expect(pds).toBe('syu.is'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('should use network config for different PDS', () => { | ||||||
|  |       const bskyConfig = getNetworkConfig('bsky.social'); | ||||||
|  |       expect(bskyConfig.pdsApi).toBe('https://bsky.social'); | ||||||
|  |       expect(bskyConfig.bskyApi).toBe('https://public.api.bsky.app'); | ||||||
|  |       expect(bskyConfig.webUrl).toBe('https://bsky.app'); | ||||||
|  |  | ||||||
|  |       const syuisConfig = getNetworkConfig('syu.is'); | ||||||
|  |       expect(syuisConfig.pdsApi).toBe('https://syu.is'); | ||||||
|  |       expect(syuisConfig.bskyApi).toBe('https://bsky.syu.is'); | ||||||
|  |       expect(syuisConfig.webUrl).toBe('https://web.syu.is'); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   describe('Environment Variable Changes', () => { | ||||||
|  |     beforeEach(() => { | ||||||
|  |       // Reset environment variables | ||||||
|  |       delete (import.meta.env as any).VITE_ATPROTO_PDS; | ||||||
|  |       delete (import.meta.env as any).VITE_ADMIN_HANDLE; | ||||||
|  |       delete (import.meta.env as any).VITE_AI_HANDLE; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('should use correct PDS for AI profile', () => { | ||||||
|  |       mockEnv({ | ||||||
|  |         VITE_ATPROTO_PDS: 'syu.is', | ||||||
|  |         VITE_ADMIN_HANDLE: 'ai.syui.ai', | ||||||
|  |         VITE_AI_HANDLE: 'ai.syui.ai' | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const config = getAppConfig(); | ||||||
|  |       expect(config.atprotoPds).toBe('syu.is'); | ||||||
|  |       expect(config.adminHandle).toBe('ai.syui.ai'); | ||||||
|  |       expect(config.aiHandle).toBe('ai.syui.ai'); | ||||||
|  |  | ||||||
|  |       // Network config should use syu.is endpoints | ||||||
|  |       const networkConfig = getNetworkConfig(config.atprotoPds); | ||||||
|  |       expect(networkConfig.bskyApi).toBe('https://bsky.syu.is'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('should construct correct API requests for admin userlist', () => { | ||||||
|  |       mockEnv({ | ||||||
|  |         VITE_ATPROTO_PDS: 'syu.is', | ||||||
|  |         VITE_ADMIN_HANDLE: 'ai.syui.ai', | ||||||
|  |         VITE_OAUTH_COLLECTION: 'ai.syui.log' | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const config = getAppConfig(); | ||||||
|  |       const networkConfig = getNetworkConfig(config.atprotoPds); | ||||||
|  |       const userListUrl = `${networkConfig.pdsApi}/xrpc/com.atproto.repo.listRecords?repo=${config.adminHandle}&collection=${config.collections.base}.user`; | ||||||
|  |        | ||||||
|  |       expect(userListUrl).toBe('https://syu.is/xrpc/com.atproto.repo.listRecords?repo=ai.syui.ai&collection=ai.syui.log.user'); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   describe('OAuth Login Flow', () => { | ||||||
|  |     it('should use syu.is OAuth for handles in VITE_ATPROTO_HANDLE_LIST', () => { | ||||||
|  |       mockEnv({ | ||||||
|  |         VITE_ATPROTO_HANDLE_LIST: '["syui.ai", "ai.syui.ai", "yui.syui.ai"]', | ||||||
|  |         VITE_ATPROTO_PDS: 'syu.is' | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const config = getAppConfig(); | ||||||
|  |       const handle = 'syui.ai'; | ||||||
|  |        | ||||||
|  |       // Check if handle is in allowed list | ||||||
|  |       expect(config.allowedHandles).toContain(handle); | ||||||
|  |        | ||||||
|  |       // Should use configured PDS for OAuth | ||||||
|  |       const expectedAuthUrl = `https://${config.atprotoPds}/oauth/authorize`; | ||||||
|  |       expect(expectedAuthUrl).toContain('syu.is'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('should use syu.is OAuth for *.syu.is handles', () => { | ||||||
|  |       const handle = 'test.syu.is'; | ||||||
|  |       const pds = detectPdsFromHandle(handle); | ||||||
|  |       expect(pds).toBe('syu.is'); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Terminal display test output | ||||||
|  | export function runTerminalTests() { | ||||||
|  |   console.log('\n=== OAuth App Tests ===\n'); | ||||||
|  |    | ||||||
|  |   // Test 1: Handle input behavior | ||||||
|  |   console.log('1. Handle Input Detection:'); | ||||||
|  |   const handles = ['syui.ai', 'syui.syu.is', 'syui.syui.ai']; | ||||||
|  |   handles.forEach(handle => { | ||||||
|  |     const pds = detectPdsFromHandle(handle); | ||||||
|  |     console.log(`   ${handle} → PDS: ${pds}`); | ||||||
|  |   }); | ||||||
|  |    | ||||||
|  |   // Test 2: Environment variable impact | ||||||
|  |   console.log('\n2. Environment Variables:'); | ||||||
|  |   const config = getAppConfig(); | ||||||
|  |   console.log(`   VITE_ATPROTO_PDS: ${config.atprotoPds}`); | ||||||
|  |   console.log(`   VITE_ADMIN_HANDLE: ${config.adminHandle}`); | ||||||
|  |   console.log(`   VITE_AI_HANDLE: ${config.aiHandle}`); | ||||||
|  |   console.log(`   VITE_OAUTH_COLLECTION: ${config.collections.base}`); | ||||||
|  |    | ||||||
|  |   // Test 3: API endpoints | ||||||
|  |   console.log('\n3. API Endpoints:'); | ||||||
|  |   const networkConfig = getNetworkConfig(config.atprotoPds); | ||||||
|  |   console.log(`   Admin PDS API: ${networkConfig.pdsApi}`); | ||||||
|  |   console.log(`   Admin Bsky API: ${networkConfig.bskyApi}`); | ||||||
|  |   console.log(`   User list URL: ${networkConfig.pdsApi}/xrpc/com.atproto.repo.listRecords?repo=${config.adminHandle}&collection=${config.collections.base}.user`); | ||||||
|  |    | ||||||
|  |   // Test 4: OAuth routing | ||||||
|  |   console.log('\n4. OAuth Routing:'); | ||||||
|  |   console.log(`   Allowed handles: ${JSON.stringify(config.allowedHandles)}`); | ||||||
|  |   console.log(`   OAuth endpoint: https://${config.atprotoPds}/oauth/authorize`); | ||||||
|  |    | ||||||
|  |   console.log('\n=== End Tests ===\n'); | ||||||
|  | } | ||||||
							
								
								
									
										24
									
								
								oauth/src/types/card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								oauth/src/types/card.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | export enum CardRarity { | ||||||
|  |   NORMAL = "normal", | ||||||
|  |   RARE = "rare", | ||||||
|  |   SUPER_RARE = "super_rare", | ||||||
|  |   KIRA = "kira", | ||||||
|  |   UNIQUE = "unique" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface Card { | ||||||
|  |   id: number; | ||||||
|  |   cp: number; | ||||||
|  |   status: CardRarity; | ||||||
|  |   skill?: string; | ||||||
|  |   owner_did: string; | ||||||
|  |   obtained_at: string; | ||||||
|  |   is_unique: boolean; | ||||||
|  |   unique_id?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface CardDrawResult { | ||||||
|  |   card: Card; | ||||||
|  |   is_new: boolean; | ||||||
|  |   animation_type: string; | ||||||
|  | } | ||||||
							
								
								
									
										138
									
								
								oauth/src/utils/oauth-endpoints.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										138
									
								
								oauth/src/utils/oauth-endpoints.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,138 @@ | |||||||
|  | /** | ||||||
|  |  * OAuth dynamic endpoint handlers | ||||||
|  |  */ | ||||||
|  | import { OAuthKeyManager, generateClientMetadata } from './oauth-keys'; | ||||||
|  |  | ||||||
|  | export class OAuthEndpointHandler { | ||||||
|  |   /** | ||||||
|  |    * Initialize OAuth endpoint handlers | ||||||
|  |    */ | ||||||
|  |   static init() { | ||||||
|  |     // Intercept requests to client-metadata.json | ||||||
|  |     this.setupClientMetadataHandler(); | ||||||
|  |      | ||||||
|  |     // Intercept requests to .well-known/jwks.json | ||||||
|  |     this.setupJWKSHandler(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private static setupClientMetadataHandler() { | ||||||
|  |     // Override fetch for client-metadata.json requests | ||||||
|  |     const originalFetch = window.fetch; | ||||||
|  |      | ||||||
|  |     window.fetch = async (input: RequestInfo | URL, init?: RequestInit) => { | ||||||
|  |       const url = typeof input === 'string' ? input : input.toString(); | ||||||
|  |        | ||||||
|  |       // Only intercept local OAuth endpoints | ||||||
|  |       try { | ||||||
|  |         const urlObj = new URL(url, window.location.origin); | ||||||
|  |          | ||||||
|  |         // Only intercept requests to the same origin | ||||||
|  |         if (urlObj.origin !== window.location.origin) { | ||||||
|  |           // Pass through external API calls unchanged | ||||||
|  |           return originalFetch(input, init); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // Handle local OAuth endpoints | ||||||
|  |         if (urlObj.pathname.endsWith('/client-metadata.json')) { | ||||||
|  |           const metadata = generateClientMetadata(); | ||||||
|  |           return new Response(JSON.stringify(metadata, null, 2), { | ||||||
|  |             headers: { | ||||||
|  |               'Content-Type': 'application/json', | ||||||
|  |               'Access-Control-Allow-Origin': '*' | ||||||
|  |             } | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         if (urlObj.pathname.endsWith('/.well-known/jwks.json')) { | ||||||
|  |           try { | ||||||
|  |             const jwks = await OAuthKeyManager.getJWKS(); | ||||||
|  |             return new Response(JSON.stringify(jwks, null, 2), { | ||||||
|  |               headers: { | ||||||
|  |                 'Content-Type': 'application/json', | ||||||
|  |                 'Access-Control-Allow-Origin': '*' | ||||||
|  |               } | ||||||
|  |             }); | ||||||
|  |           } catch (error) { | ||||||
|  |             return new Response(JSON.stringify({ error: 'Failed to generate JWKS' }), { | ||||||
|  |               status: 500, | ||||||
|  |               headers: { 'Content-Type': 'application/json' } | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } catch (e) { | ||||||
|  |         // If URL parsing fails, pass through to original fetch | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       // Pass through all other requests | ||||||
|  |       return originalFetch(input, init); | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private static setupJWKSHandler() { | ||||||
|  |     // This is handled in the fetch override above | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Generate a proper client assertion JWT for token requests | ||||||
|  |    */ | ||||||
|  |   static async generateClientAssertion(tokenEndpoint: string): Promise<string> { | ||||||
|  |     const now = Math.floor(Date.now() / 1000); | ||||||
|  |     const clientId = generateClientMetadata().client_id; | ||||||
|  |  | ||||||
|  |     const header = { | ||||||
|  |       alg: 'ES256', | ||||||
|  |       typ: 'JWT', | ||||||
|  |       kid: 'ai-card-oauth-key-1' | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const payload = { | ||||||
|  |       iss: clientId, | ||||||
|  |       sub: clientId, | ||||||
|  |       aud: tokenEndpoint, | ||||||
|  |       iat: now, | ||||||
|  |       exp: now + 300, // 5 minutes | ||||||
|  |       jti: crypto.randomUUID() | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     return await OAuthKeyManager.signJWT(header, payload); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Service Worker alternative for intercepting requests | ||||||
|  |  * (This is a more robust solution for production) | ||||||
|  |  */ | ||||||
|  | export function registerOAuthServiceWorker() { | ||||||
|  |   if ('serviceWorker' in navigator) { | ||||||
|  |     const swCode = ` | ||||||
|  |       self.addEventListener('fetch', (event) => { | ||||||
|  |         const url = new URL(event.request.url); | ||||||
|  |          | ||||||
|  |         if (url.pathname.endsWith('/client-metadata.json')) { | ||||||
|  |           event.respondWith( | ||||||
|  |             new Response(JSON.stringify({ | ||||||
|  |               client_id: url.origin + '/client-metadata.json', | ||||||
|  |               client_name: 'ai.card', | ||||||
|  |               client_uri: url.origin, | ||||||
|  |               redirect_uris: [url.origin + '/oauth/callback'], | ||||||
|  |               response_types: ['code'], | ||||||
|  |               grant_types: ['authorization_code', 'refresh_token'], | ||||||
|  |               token_endpoint_auth_method: 'private_key_jwt', | ||||||
|  |               scope: 'atproto transition:generic', | ||||||
|  |               subject_type: 'public', | ||||||
|  |               application_type: 'web', | ||||||
|  |               dpop_bound_access_tokens: true, | ||||||
|  |               jwks_uri: url.origin + '/.well-known/jwks.json' | ||||||
|  |             }, null, 2), { | ||||||
|  |               headers: { 'Content-Type': 'application/json' } | ||||||
|  |             }) | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     `; | ||||||
|  |      | ||||||
|  |     const blob = new Blob([swCode], { type: 'application/javascript' }); | ||||||
|  |     const swUrl = URL.createObjectURL(blob); | ||||||
|  |      | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										181
									
								
								oauth/src/utils/oauth-keys.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										181
									
								
								oauth/src/utils/oauth-keys.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,181 @@ | |||||||
|  | /** | ||||||
|  |  * OAuth JWKS key generation and management | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | export interface JWK { | ||||||
|  |   kty: string; | ||||||
|  |   crv: string; | ||||||
|  |   x: string; | ||||||
|  |   y: string; | ||||||
|  |   d?: string; | ||||||
|  |   use: string; | ||||||
|  |   kid: string; | ||||||
|  |   alg: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface JWKS { | ||||||
|  |   keys: JWK[]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export class OAuthKeyManager { | ||||||
|  |   private static keyPair: CryptoKeyPair | null = null; | ||||||
|  |   private static jwks: JWKS | null = null; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Generate or retrieve existing ECDSA key pair for OAuth | ||||||
|  |    */ | ||||||
|  |   static async getKeyPair(): Promise<CryptoKeyPair> { | ||||||
|  |     if (this.keyPair) { | ||||||
|  |       return this.keyPair; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Try to load from localStorage first | ||||||
|  |     const storedKey = localStorage.getItem('oauth_private_key'); | ||||||
|  |     if (storedKey) { | ||||||
|  |       try { | ||||||
|  |         const keyData = JSON.parse(storedKey); | ||||||
|  |         this.keyPair = await this.importKeyPair(keyData); | ||||||
|  |         return this.keyPair; | ||||||
|  |       } catch (error) { | ||||||
|  |         localStorage.removeItem('oauth_private_key'); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Generate new key pair | ||||||
|  |     this.keyPair = await window.crypto.subtle.generateKey( | ||||||
|  |       { | ||||||
|  |         name: 'ECDSA', | ||||||
|  |         namedCurve: 'P-256', | ||||||
|  |       }, | ||||||
|  |       true, // extractable | ||||||
|  |       ['sign', 'verify'] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // Store private key for persistence | ||||||
|  |     await this.storeKeyPair(this.keyPair); | ||||||
|  |  | ||||||
|  |     return this.keyPair; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Get JWKS (JSON Web Key Set) for public key distribution | ||||||
|  |    */ | ||||||
|  |   static async getJWKS(): Promise<JWKS> { | ||||||
|  |     if (this.jwks) { | ||||||
|  |       return this.jwks; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const keyPair = await this.getKeyPair(); | ||||||
|  |     const publicKey = await window.crypto.subtle.exportKey('jwk', keyPair.publicKey); | ||||||
|  |  | ||||||
|  |     this.jwks = { | ||||||
|  |       keys: [ | ||||||
|  |         { | ||||||
|  |           kty: publicKey.kty!, | ||||||
|  |           crv: publicKey.crv!, | ||||||
|  |           x: publicKey.x!, | ||||||
|  |           y: publicKey.y!, | ||||||
|  |           use: 'sig', | ||||||
|  |           kid: 'ai-card-oauth-key-1', | ||||||
|  |           alg: 'ES256' | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     return this.jwks; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Sign a JWT with the private key | ||||||
|  |    */ | ||||||
|  |   static async signJWT(header: any, payload: any): Promise<string> { | ||||||
|  |     const keyPair = await this.getKeyPair(); | ||||||
|  |      | ||||||
|  |     const headerB64 = btoa(JSON.stringify(header)).replace(/=/g, ''); | ||||||
|  |     const payloadB64 = btoa(JSON.stringify(payload)).replace(/=/g, ''); | ||||||
|  |     const message = `${headerB64}.${payloadB64}`; | ||||||
|  |      | ||||||
|  |     const signature = await window.crypto.subtle.sign( | ||||||
|  |       { name: 'ECDSA', hash: 'SHA-256' }, | ||||||
|  |       keyPair.privateKey, | ||||||
|  |       new TextEncoder().encode(message) | ||||||
|  |     ); | ||||||
|  |      | ||||||
|  |     const signatureB64 = btoa(String.fromCharCode(...new Uint8Array(signature))) | ||||||
|  |       .replace(/\+/g, '-') | ||||||
|  |       .replace(/\//g, '_') | ||||||
|  |       .replace(/=/g, ''); | ||||||
|  |      | ||||||
|  |     return `${message}.${signatureB64}`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private static async storeKeyPair(keyPair: CryptoKeyPair): Promise<void> { | ||||||
|  |     try { | ||||||
|  |       const privateKey = await window.crypto.subtle.exportKey('jwk', keyPair.privateKey); | ||||||
|  |       localStorage.setItem('oauth_private_key', JSON.stringify(privateKey)); | ||||||
|  |     } catch (error) { | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private static async importKeyPair(keyData: any): Promise<CryptoKeyPair> { | ||||||
|  |     const privateKey = await window.crypto.subtle.importKey( | ||||||
|  |       'jwk', | ||||||
|  |       keyData, | ||||||
|  |       { name: 'ECDSA', namedCurve: 'P-256' }, | ||||||
|  |       true, | ||||||
|  |       ['sign'] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // Derive public key from private key | ||||||
|  |     const publicKeyData = { ...keyData }; | ||||||
|  |     delete publicKeyData.d; // Remove private component | ||||||
|  |  | ||||||
|  |     const publicKey = await window.crypto.subtle.importKey( | ||||||
|  |       'jwk', | ||||||
|  |       publicKeyData, | ||||||
|  |       { name: 'ECDSA', namedCurve: 'P-256' }, | ||||||
|  |       true, | ||||||
|  |       ['verify'] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     return { privateKey, publicKey }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Clear stored keys (for testing/reset) | ||||||
|  |    */ | ||||||
|  |   static clearKeys(): void { | ||||||
|  |     localStorage.removeItem('oauth_private_key'); | ||||||
|  |     this.keyPair = null; | ||||||
|  |     this.jwks = null; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Generate dynamic client metadata based on current URL | ||||||
|  |  */ | ||||||
|  | export function generateClientMetadata(): any { | ||||||
|  |   // Use environment variables if available, fallback to current origin | ||||||
|  |   const host = import.meta.env.VITE_APP_HOST || window.location.origin; | ||||||
|  |   const clientId = import.meta.env.VITE_OAUTH_CLIENT_ID || `${host}/client-metadata.json`; | ||||||
|  |   const redirectUri = import.meta.env.VITE_OAUTH_REDIRECT_URI || `${host}/oauth/callback`; | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     client_id: clientId, | ||||||
|  |     client_name: 'ai.card', | ||||||
|  |     client_uri: host, | ||||||
|  |     logo_uri: `${host}/favicon.ico`, | ||||||
|  |     tos_uri: `${host}/terms`, | ||||||
|  |     policy_uri: `${host}/privacy`, | ||||||
|  |     redirect_uris: [redirectUri, host], | ||||||
|  |     response_types: ['code'], | ||||||
|  |     grant_types: ['authorization_code', 'refresh_token'], | ||||||
|  |     token_endpoint_auth_method: 'private_key_jwt', | ||||||
|  |     token_endpoint_auth_signing_alg: 'ES256', | ||||||
|  |     scope: 'atproto transition:generic', | ||||||
|  |     subject_type: 'public', | ||||||
|  |     application_type: 'web', | ||||||
|  |     dpop_bound_access_tokens: true, | ||||||
|  |     jwks_uri: `${host}/.well-known/jwks.json` | ||||||
|  |   }; | ||||||
|  | } | ||||||
							
								
								
									
										348
									
								
								oauth/src/utils/pds-detection.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										348
									
								
								oauth/src/utils/pds-detection.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,348 @@ | |||||||
|  | // PDS Detection and API URL mapping utilities | ||||||
|  |  | ||||||
|  | import { isValidDid, isValidHandle } from './validation'; | ||||||
|  |  | ||||||
|  | export interface NetworkConfig { | ||||||
|  |   pdsApi: string; | ||||||
|  |   plcApi: string; | ||||||
|  |   bskyApi: string; | ||||||
|  |   webUrl: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Detect PDS from handle | ||||||
|  | export function detectPdsFromHandle(handle: string): string { | ||||||
|  |   // Get allowed handles from environment | ||||||
|  |   const allowedHandlesStr = import.meta.env.VITE_ATPROTO_HANDLE_LIST || '[]'; | ||||||
|  |   let allowedHandles: string[] = []; | ||||||
|  |   try { | ||||||
|  |     allowedHandles = JSON.parse(allowedHandlesStr); | ||||||
|  |   } catch { | ||||||
|  |     allowedHandles = []; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // Get configured PDS from environment | ||||||
|  |   const configuredPds = import.meta.env.VITE_ATPROTO_PDS || 'syu.is'; | ||||||
|  |    | ||||||
|  |   // Check if handle is in allowed list | ||||||
|  |   if (allowedHandles.includes(handle)) { | ||||||
|  |     return configuredPds; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // Check if handle ends with .syu.is or .syui.ai | ||||||
|  |   if (handle.endsWith('.syu.is') || handle.endsWith('.syui.ai')) { | ||||||
|  |     return 'syu.is'; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // Check if handle ends with .bsky.social or .bsky.app | ||||||
|  |   if (handle.endsWith('.bsky.social') || handle.endsWith('.bsky.app')) { | ||||||
|  |     return 'bsky.social'; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // Default to Bluesky for unknown domains | ||||||
|  |   return 'bsky.social'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Map PDS endpoint to network configuration | ||||||
|  | export function getNetworkConfigFromPdsEndpoint(pdsEndpoint: string): NetworkConfig { | ||||||
|  |   try { | ||||||
|  |     const url = new URL(pdsEndpoint); | ||||||
|  |     const hostname = url.hostname; | ||||||
|  |      | ||||||
|  |     // Map based on actual PDS endpoint | ||||||
|  |     if (hostname === 'syu.is') { | ||||||
|  |       return { | ||||||
|  |         pdsApi: 'https://syu.is',           // PDS API (repo operations) | ||||||
|  |         plcApi: 'https://plc.syu.is',       // PLC directory | ||||||
|  |         bskyApi: 'https://bsky.syu.is',     // Bluesky API (getProfile, etc.) | ||||||
|  |         webUrl: 'https://web.syu.is'        // Web interface | ||||||
|  |       }; | ||||||
|  |     } else if (hostname.includes('bsky.network') || hostname === 'bsky.social' || hostname.includes('host.bsky.network')) { | ||||||
|  |       // All Bluesky infrastructure (including *.host.bsky.network) | ||||||
|  |       return { | ||||||
|  |         pdsApi: pdsEndpoint,                      // Use actual PDS endpoint (e.g., shiitake.us-east.host.bsky.network) | ||||||
|  |         plcApi: 'https://plc.directory',          // Standard PLC directory | ||||||
|  |         bskyApi: 'https://public.api.bsky.app',   // Bluesky public API (NOT PDS) | ||||||
|  |         webUrl: 'https://bsky.app'                // Bluesky web interface | ||||||
|  |       }; | ||||||
|  |     } else { | ||||||
|  |       // Unknown PDS, assume Bluesky-compatible but use PDS for repo operations | ||||||
|  |       return { | ||||||
|  |         pdsApi: pdsEndpoint,                      // Use actual PDS for repo ops | ||||||
|  |         plcApi: 'https://plc.directory',          // Default PLC | ||||||
|  |         bskyApi: 'https://public.api.bsky.app',   // Default to Bluesky API | ||||||
|  |         webUrl: 'https://bsky.app'                // Default web interface | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     // Fallback for invalid URLs | ||||||
|  |     return { | ||||||
|  |       pdsApi: 'https://bsky.social', | ||||||
|  |       plcApi: 'https://plc.directory', | ||||||
|  |       bskyApi: 'https://public.api.bsky.app', | ||||||
|  |       webUrl: 'https://bsky.app' | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Legacy function for backwards compatibility | ||||||
|  | export function getNetworkConfig(pds: string): NetworkConfig { | ||||||
|  |   // This now assumes pds is a hostname | ||||||
|  |   return getNetworkConfigFromPdsEndpoint(`https://${pds}`); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Get appropriate API URL for a user based on their handle | ||||||
|  | export function getApiUrlForUser(handle: string): string { | ||||||
|  |   const pds = detectPdsFromHandle(handle); | ||||||
|  |   const config = getNetworkConfig(pds); | ||||||
|  |   return config.bskyApi; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Resolve handle/DID to actual PDS endpoint using PLC API first | ||||||
|  | export async function resolvePdsFromRepo(handleOrDid: string): Promise<{ pds: string; did: string; handle: string }> { | ||||||
|  |   // Validate input | ||||||
|  |   if (!handleOrDid || (!isValidDid(handleOrDid) && !isValidHandle(handleOrDid))) { | ||||||
|  |     throw new Error(`Invalid identifier: ${handleOrDid}`); | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   let targetDid = handleOrDid; | ||||||
|  |   let targetHandle = handleOrDid; | ||||||
|  |    | ||||||
|  |   // If handle provided, resolve to DID first using identity.resolveHandle | ||||||
|  |   if (!handleOrDid.startsWith('did:')) { | ||||||
|  |     try { | ||||||
|  |       // Try multiple endpoints for handle resolution | ||||||
|  |       const resolveEndpoints = ['https://public.api.bsky.app', 'https://bsky.syu.is', 'https://syu.is']; | ||||||
|  |       let resolved = false; | ||||||
|  |        | ||||||
|  |       for (const endpoint of resolveEndpoints) { | ||||||
|  |         try { | ||||||
|  |           const resolveResponse = await fetch(`${endpoint}/xrpc/com.atproto.identity.resolveHandle?handle=${encodeURIComponent(handleOrDid)}`); | ||||||
|  |           if (resolveResponse.ok) { | ||||||
|  |             const resolveData = await resolveResponse.json(); | ||||||
|  |             targetDid = resolveData.did; | ||||||
|  |             resolved = true; | ||||||
|  |             break; | ||||||
|  |           } | ||||||
|  |         } catch (error) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       if (!resolved) { | ||||||
|  |         throw new Error('Handle resolution failed from all endpoints'); | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       throw new Error(`Failed to resolve handle ${handleOrDid} to DID: ${error}`); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // First, try PLC API to get the authoritative DID document | ||||||
|  |   const plcApis = ['https://plc.directory', 'https://plc.syu.is']; | ||||||
|  |    | ||||||
|  |   for (const plcApi of plcApis) { | ||||||
|  |     try { | ||||||
|  |       const plcResponse = await fetch(`${plcApi}/${targetDid}`); | ||||||
|  |       if (plcResponse.ok) { | ||||||
|  |         const didDocument = await plcResponse.json(); | ||||||
|  |          | ||||||
|  |         // Find PDS service in DID document | ||||||
|  |         const pdsService = didDocument.service?.find((s: any) =>  | ||||||
|  |           s.id === '#atproto_pds' || s.type === 'AtprotoPersonalDataServer' | ||||||
|  |         ); | ||||||
|  |          | ||||||
|  |         if (pdsService && pdsService.serviceEndpoint) { | ||||||
|  |           return { | ||||||
|  |             pds: pdsService.serviceEndpoint, | ||||||
|  |             did: targetDid, | ||||||
|  |             handle: targetHandle | ||||||
|  |           }; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // Fallback: use com.atproto.repo.describeRepo to get PDS from known PDS endpoints | ||||||
|  |   const pdsEndpoints = ['https://bsky.social', 'https://syu.is']; | ||||||
|  |    | ||||||
|  |   for (const pdsEndpoint of pdsEndpoints) { | ||||||
|  |     try { | ||||||
|  |       const response = await fetch(`${pdsEndpoint}/xrpc/com.atproto.repo.describeRepo?repo=${encodeURIComponent(targetDid)}`); | ||||||
|  |        | ||||||
|  |       if (response.ok) { | ||||||
|  |         const data = await response.json(); | ||||||
|  |          | ||||||
|  |         // Extract PDS from didDoc.service | ||||||
|  |         const services = data.didDoc?.service || []; | ||||||
|  |         const pdsService = services.find((s: any) =>  | ||||||
|  |           s.id === '#atproto_pds' || s.type === 'AtprotoPersonalDataServer' | ||||||
|  |         ); | ||||||
|  |          | ||||||
|  |         if (pdsService) { | ||||||
|  |           return { | ||||||
|  |             pds: pdsService.serviceEndpoint, | ||||||
|  |             did: data.did || targetDid, | ||||||
|  |             handle: data.handle || targetHandle | ||||||
|  |           }; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   throw new Error(`Failed to resolve PDS for ${handleOrDid} from any endpoint`); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Resolve DID to actual PDS endpoint using com.atproto.repo.describeRepo | ||||||
|  | export async function resolvePdsFromDid(did: string): Promise<string> { | ||||||
|  |   const resolved = await resolvePdsFromRepo(did); | ||||||
|  |   return resolved.pds; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Enhanced resolve handle to DID with proper PDS detection | ||||||
|  | export async function resolveHandleToDid(handle: string): Promise<{ did: string; pds: string }> { | ||||||
|  |   try { | ||||||
|  |     // First, try to resolve the handle to DID using multiple methods | ||||||
|  |     const apiUrl = getApiUrlForUser(handle); | ||||||
|  |     const response = await fetch(`${apiUrl}/xrpc/app.bsky.actor.getProfile?actor=${encodeURIComponent(handle)}`); | ||||||
|  |      | ||||||
|  |     if (!response.ok) { | ||||||
|  |       throw new Error(`Failed to resolve handle: ${response.status}`); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const data = await response.json(); | ||||||
|  |     const did = data.did; | ||||||
|  |      | ||||||
|  |     // Now resolve the actual PDS from the DID | ||||||
|  |     const actualPds = await resolvePdsFromDid(did); | ||||||
|  |      | ||||||
|  |     return { | ||||||
|  |       did: did, | ||||||
|  |       pds: actualPds | ||||||
|  |     }; | ||||||
|  |   } catch (error) { | ||||||
|  |     // Failed to resolve handle | ||||||
|  |      | ||||||
|  |     // Fallback to handle-based detection | ||||||
|  |     const fallbackPds = detectPdsFromHandle(handle); | ||||||
|  |     throw error; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Get profile using appropriate API for the user with accurate PDS resolution | ||||||
|  | export async function getProfileForUser(handleOrDid: string, knownPdsEndpoint?: string): Promise<any> { | ||||||
|  |   try { | ||||||
|  |     let apiUrl: string; | ||||||
|  |      | ||||||
|  |     if (knownPdsEndpoint) { | ||||||
|  |       // If we already know the user's PDS endpoint, use it directly | ||||||
|  |       const config = getNetworkConfigFromPdsEndpoint(knownPdsEndpoint); | ||||||
|  |       apiUrl = config.bskyApi; | ||||||
|  |     } else { | ||||||
|  |       // Resolve the user's actual PDS using describeRepo | ||||||
|  |       try { | ||||||
|  |         const resolved = await resolvePdsFromRepo(handleOrDid); | ||||||
|  |         const config = getNetworkConfigFromPdsEndpoint(resolved.pds); | ||||||
|  |         apiUrl = config.bskyApi; | ||||||
|  |       } catch { | ||||||
|  |         // Fallback to handle-based detection | ||||||
|  |         apiUrl = getApiUrlForUser(handleOrDid); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     const response = await fetch(`${apiUrl}/xrpc/app.bsky.actor.getProfile?actor=${encodeURIComponent(handleOrDid)}`); | ||||||
|  |     if (!response.ok) { | ||||||
|  |       throw new Error(`Failed to get profile: ${response.status}`); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     return await response.json(); | ||||||
|  |   } catch (error) { | ||||||
|  |     // Failed to get profile | ||||||
|  |      | ||||||
|  |     // Final fallback: try with default Bluesky API | ||||||
|  |     try { | ||||||
|  |       const response = await fetch(`https://public.api.bsky.app/xrpc/app.bsky.actor.getProfile?actor=${encodeURIComponent(handleOrDid)}`); | ||||||
|  |       if (response.ok) { | ||||||
|  |         return await response.json(); | ||||||
|  |       } | ||||||
|  |     } catch { | ||||||
|  |       // Ignore fallback errors | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     throw error; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Test and verify PDS detection methods | ||||||
|  | export async function verifyPdsDetection(handleOrDid: string): Promise<void> { | ||||||
|  |   try { | ||||||
|  |     // Method 1: com.atproto.repo.describeRepo (PRIMARY METHOD) | ||||||
|  |     try { | ||||||
|  |       const resolved = await resolvePdsFromRepo(handleOrDid); | ||||||
|  |       const config = getNetworkConfigFromPdsEndpoint(resolved.pds); | ||||||
|  |     } catch (error) { | ||||||
|  |       // describeRepo failed | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Method 2: com.atproto.identity.resolveHandle (for comparison) | ||||||
|  |     if (!handleOrDid.startsWith('did:')) { | ||||||
|  |       try { | ||||||
|  |         const resolveResponse = await fetch(`https://public.api.bsky.app/xrpc/com.atproto.identity.resolveHandle?handle=${encodeURIComponent(handleOrDid)}`); | ||||||
|  |         if (resolveResponse.ok) { | ||||||
|  |           const resolveData = await resolveResponse.json(); | ||||||
|  |         } | ||||||
|  |       } catch (error) { | ||||||
|  |         // Error resolving handle | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Method 3: PLC Directory lookup (if we have a DID) | ||||||
|  |     let targetDid = handleOrDid; | ||||||
|  |     if (!handleOrDid.startsWith('did:')) { | ||||||
|  |       try { | ||||||
|  |         const profile = await getProfileForUser(handleOrDid); | ||||||
|  |         targetDid = profile.did; | ||||||
|  |       } catch { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       const plcResponse = await fetch(`https://plc.directory/${targetDid}`); | ||||||
|  |       if (plcResponse.ok) { | ||||||
|  |         const didDocument = await plcResponse.json(); | ||||||
|  |          | ||||||
|  |         // Find PDS service | ||||||
|  |         const pdsService = didDocument.service?.find((s: any) =>  | ||||||
|  |           s.id === '#atproto_pds' || s.type === 'AtprotoPersonalDataServer' | ||||||
|  |         ); | ||||||
|  |          | ||||||
|  |         if (pdsService) { | ||||||
|  |           // Try to detect if this is a known network | ||||||
|  |           const pdsUrl = pdsService.serviceEndpoint; | ||||||
|  |           const hostname = new URL(pdsUrl).hostname; | ||||||
|  |           const detectedNetwork = detectPdsFromHandle(`user.${hostname}`); | ||||||
|  |           const networkConfig = getNetworkConfig(hostname); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       // Error fetching from PLC directory | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Method 4: Our enhanced resolution | ||||||
|  |     try { | ||||||
|  |       if (handleOrDid.startsWith('did:')) { | ||||||
|  |         const pdsEndpoint = await resolvePdsFromDid(handleOrDid); | ||||||
|  |       } else { | ||||||
|  |         const resolved = await resolveHandleToDid(handleOrDid); | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       // Enhanced resolution failed | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |   } catch (error) { | ||||||
|  |     // Overall verification failed | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										21
									
								
								oauth/src/utils/validation.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								oauth/src/utils/validation.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | // Validation utilities for atproto identifiers | ||||||
|  |  | ||||||
|  | export function isValidDid(did: string): boolean { | ||||||
|  |   if (!did || typeof did !== 'string') return false; | ||||||
|  |    | ||||||
|  |   // Basic DID format: did:method:identifier | ||||||
|  |   const didRegex = /^did:[a-z]+:[a-zA-Z0-9._%-]+$/; | ||||||
|  |   return didRegex.test(did); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function isValidHandle(handle: string): boolean { | ||||||
|  |   if (!handle || typeof handle !== 'string') return false; | ||||||
|  |    | ||||||
|  |   // Basic handle format: subdomain.domain.tld | ||||||
|  |   const handleRegex = /^([a-zA-Z0-9]([a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?\.)+[a-zA-Z]{2,}$/; | ||||||
|  |   return handleRegex.test(handle); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function isValidAtprotoIdentifier(identifier: string): boolean { | ||||||
|  |   return isValidDid(identifier) || isValidHandle(identifier); | ||||||
|  | } | ||||||
							
								
								
									
										21
									
								
								oauth/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								oauth/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | { | ||||||
|  |   "compilerOptions": { | ||||||
|  |     "target": "ES2020", | ||||||
|  |     "useDefineForClassFields": true, | ||||||
|  |     "lib": ["ES2020", "DOM", "DOM.Iterable"], | ||||||
|  |     "module": "ESNext", | ||||||
|  |     "skipLibCheck": true, | ||||||
|  |     "moduleResolution": "bundler", | ||||||
|  |     "allowImportingTsExtensions": true, | ||||||
|  |     "resolveJsonModule": true, | ||||||
|  |     "isolatedModules": true, | ||||||
|  |     "noEmit": true, | ||||||
|  |     "jsx": "react-jsx", | ||||||
|  |     "strict": true, | ||||||
|  |     "noUnusedLocals": true, | ||||||
|  |     "noUnusedParameters": true, | ||||||
|  |     "noFallthroughCasesInSwitch": true | ||||||
|  |   }, | ||||||
|  |   "include": ["src"], | ||||||
|  |   "references": [{ "path": "./tsconfig.node.json" }] | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								oauth/tsconfig.node.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								oauth/tsconfig.node.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | { | ||||||
|  |   "compilerOptions": { | ||||||
|  |     "composite": true, | ||||||
|  |     "skipLibCheck": true, | ||||||
|  |     "module": "ESNext", | ||||||
|  |     "moduleResolution": "bundler", | ||||||
|  |     "allowSyntheticDefaultImports": true, | ||||||
|  |     "strict": true | ||||||
|  |   }, | ||||||
|  |   "include": ["vite.config.ts"] | ||||||
|  | } | ||||||
							
								
								
									
										88
									
								
								oauth/vite.config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								oauth/vite.config.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | |||||||
|  | import { defineConfig, loadEnv } from 'vite' | ||||||
|  | import react from '@vitejs/plugin-react' | ||||||
|  | import fs from 'fs' | ||||||
|  | import path from 'path' | ||||||
|  |  | ||||||
|  | export default defineConfig(({ mode }) => { | ||||||
|  |   // Load env file based on `mode` in the current working directory. | ||||||
|  |   const env = loadEnv(mode, process.cwd(), '') | ||||||
|  |    | ||||||
|  |   return { | ||||||
|  |     plugins: [ | ||||||
|  |       react(), | ||||||
|  |       // Custom plugin to replace variables in public files during build | ||||||
|  |       { | ||||||
|  |         name: 'replace-env-vars', | ||||||
|  |         writeBundle() { | ||||||
|  |           const host = env.VITE_APP_HOST || 'https://log.syui.ai' | ||||||
|  |           const clientId = env.VITE_OAUTH_CLIENT_ID || `${host}/client-metadata.json` | ||||||
|  |           const redirectUri = env.VITE_OAUTH_REDIRECT_URI || `${host}/oauth/callback` | ||||||
|  |            | ||||||
|  |           // Replace variables in client-metadata.json | ||||||
|  |           const clientMetadataPath = path.resolve(__dirname, 'dist/client-metadata.json') | ||||||
|  |           if (fs.existsSync(clientMetadataPath)) { | ||||||
|  |             let content = fs.readFileSync(clientMetadataPath, 'utf-8') | ||||||
|  |             content = content.replace(/https:\/\/log\.syui\.ai/g, host) | ||||||
|  |             fs.writeFileSync(clientMetadataPath, content) | ||||||
|  |             console.log(`Updated client-metadata.json with host: ${host}`) | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       // Generate standalone index.html for testing | ||||||
|  |       { | ||||||
|  |         name: 'generate-standalone-html', | ||||||
|  |         writeBundle(options, bundle) { | ||||||
|  |           // Find actual generated filenames | ||||||
|  |           const jsFile = Object.keys(bundle).find(fileName => fileName.startsWith('assets/comment-atproto') && fileName.endsWith('.js')) | ||||||
|  |           const cssFile = Object.keys(bundle).find(fileName => fileName.startsWith('assets/comment-atproto') && fileName.endsWith('.css')) | ||||||
|  |            | ||||||
|  |           // Generate minimal index.html with just asset references | ||||||
|  |           const indexHtmlPath = path.resolve(__dirname, 'dist/index.html') | ||||||
|  |           const indexHtmlContent = `<!-- OAuth Comment System - Load globally for session management --> | ||||||
|  | <script type="module" crossorigin src="/${jsFile}"></script> | ||||||
|  | <link rel="stylesheet" crossorigin href="/${cssFile}">` | ||||||
|  |           fs.writeFileSync(indexHtmlPath, indexHtmlContent) | ||||||
|  |           console.log('Generated minimal index.html with asset references') | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     build: { | ||||||
|  |       // Keep console.log in production for debugging | ||||||
|  |       minify: 'esbuild', | ||||||
|  |       rollupOptions: { | ||||||
|  |         output: { | ||||||
|  |           // Hash-based filenames to bust cache | ||||||
|  |           entryFileNames: 'assets/comment-atproto-[hash].js', | ||||||
|  |           chunkFileNames: 'assets/comment-atproto-[name]-[hash].js', | ||||||
|  |           assetFileNames: (assetInfo) => { | ||||||
|  |             if (assetInfo.name && assetInfo.name.endsWith('.css')) { | ||||||
|  |               return 'assets/comment-atproto-[hash].css'; | ||||||
|  |             } | ||||||
|  |             return 'assets/[name]-[hash].[ext]'; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     esbuild: { | ||||||
|  |       drop: [], // Don't drop console.log | ||||||
|  |     }, | ||||||
|  |     server: { | ||||||
|  |       port: 5173, | ||||||
|  |       host: '127.0.0.1', | ||||||
|  |       allowedHosts: ['localhost', '127.0.0.1', 'log.syui.ai'], | ||||||
|  |       proxy: { | ||||||
|  |         '/api': { | ||||||
|  |           target: 'http://127.0.0.1:8000', | ||||||
|  |           changeOrigin: true, | ||||||
|  |           secure: false, | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       // Handle OAuth callback routing | ||||||
|  |       historyApiFallback: { | ||||||
|  |         rewrites: [ | ||||||
|  |           { from: /^\/oauth\/callback/, to: '/index.html' } | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }) | ||||||
							
								
								
									
										6
									
								
								oauth_new/.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								oauth_new/.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | VITE_ADMIN=ai.syui.ai | ||||||
|  | VITE_PDS=syu.is | ||||||
|  | VITE_HANDLE_LIST=["ai.syui.ai", "syui.syui.ai", "ai.ai"] | ||||||
|  | VITE_COLLECTION=ai.syui.log | ||||||
|  | VITE_OAUTH_CLIENT_ID=https://syui.ai/client-metadata.json | ||||||
|  | VITE_OAUTH_REDIRECT_URI=https://syui.ai/oauth/callback | ||||||
							
								
								
									
										334
									
								
								oauth_new/DEVELOPMENT.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										334
									
								
								oauth_new/DEVELOPMENT.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,334 @@ | |||||||
|  | # 開発ガイド | ||||||
|  |  | ||||||
|  | ## 設計思想 | ||||||
|  |  | ||||||
|  | このプロジェクトは以下の原則に基づいて設計されています: | ||||||
|  |  | ||||||
|  | ### 1. 環境変数による設定の外部化 | ||||||
|  | - ハードコードを避け、設定は全て環境変数で管理 | ||||||
|  | - `src/config/env.js` で一元管理 | ||||||
|  |  | ||||||
|  | ### 2. PDS(Personal Data Server)の自動判定 | ||||||
|  | - `VITE_HANDLE_LIST` と `VITE_PDS` による自動判定 | ||||||
|  | - syu.is系とbsky.social系の自動振り分け | ||||||
|  |  | ||||||
|  | ### 3. コンポーネントの責任分離 | ||||||
|  | - Hooks: ビジネスロジック | ||||||
|  | - Components: UI表示のみ | ||||||
|  | - Services: 外部API連携 | ||||||
|  | - Utils: 純粋関数 | ||||||
|  |  | ||||||
|  | ## アーキテクチャ詳細 | ||||||
|  |  | ||||||
|  | ### データフロー | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | User Input | ||||||
|  |     ↓ | ||||||
|  | Hooks (useAuth, useAdminData, usePageContext) | ||||||
|  |     ↓ | ||||||
|  | Services (OAuthService) | ||||||
|  |     ↓ | ||||||
|  | API (atproto.js) | ||||||
|  |     ↓ | ||||||
|  | ATProto Network | ||||||
|  |     ↓ | ||||||
|  | Components (UI Display) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 状態管理 | ||||||
|  |  | ||||||
|  | React Hooksによる状態管理: | ||||||
|  | - `useAuth`: OAuth認証状態 | ||||||
|  | - `useAdminData`: 管理者データ(プロフィール、レコード) | ||||||
|  | - `usePageContext`: ページ判定(トップ/個別) | ||||||
|  |  | ||||||
|  | ### OAuth認証フロー | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | 1. ユーザーがハンドル入力 | ||||||
|  | 2. PDS判定 (syu.is vs bsky.social) | ||||||
|  | 3. 適切なOAuthClientを選択 | ||||||
|  | 4. 標準OAuth画面にリダイレクト | ||||||
|  | 5. 認証完了後コールバック処理 | ||||||
|  | 6. セッション復元・保存 | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 重要な実装詳細 | ||||||
|  |  | ||||||
|  | ### セッション管理 | ||||||
|  |  | ||||||
|  | `@atproto/oauth-client-browser`が自動的に以下を処理: | ||||||
|  | - IndexedDBへのセッション保存 | ||||||
|  | - トークンの自動更新 | ||||||
|  | - DPoP(Demonstration of Proof of Possession) | ||||||
|  |  | ||||||
|  | **注意**: 手動でのセッション管理は複雑なため、公式ライブラリを使用すること。 | ||||||
|  |  | ||||||
|  | ### PDS判定アルゴリズム | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/utils/pds.js | ||||||
|  | function isSyuIsHandle(handle) { | ||||||
|  |   return env.handleList.includes(handle) || handle.endsWith(`.${env.pds}`) | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 1. `VITE_HANDLE_LIST` に含まれるハンドル → syu.is | ||||||
|  | 2. `.syu.is` で終わるハンドル → syu.is   | ||||||
|  | 3. その他 → bsky.social | ||||||
|  |  | ||||||
|  | ### レコードフィルタリング | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/components/RecordTabs.jsx | ||||||
|  | const filterRecords = (records) => { | ||||||
|  |   if (pageContext.isTopPage) { | ||||||
|  |     return records.slice(0, 3) // 最新3件 | ||||||
|  |   } else { | ||||||
|  |     // URL のrkey と record.value.post.url のrkey を照合 | ||||||
|  |     return records.filter(record => { | ||||||
|  |       const recordRkey = new URL(record.value.post.url).pathname.split('/').pop()?.replace(/\.html$/, '') | ||||||
|  |       return recordRkey === pageContext.rkey | ||||||
|  |     }) | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 開発時の注意点 | ||||||
|  |  | ||||||
|  | ### 1. 環境変数の命名 | ||||||
|  |  | ||||||
|  | - `VITE_` プレフィックス必須(Viteの制約) | ||||||
|  | - JSON形式の環境変数は文字列として定義 | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # ❌ 間違い | ||||||
|  | VITE_HANDLE_LIST=["ai.syui.ai"] | ||||||
|  |  | ||||||
|  | # ✅ 正しい   | ||||||
|  | VITE_HANDLE_LIST=["ai.syui.ai", "syui.syui.ai"] | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 2. API エラーハンドリング | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/api/atproto.js | ||||||
|  | async function request(url) { | ||||||
|  |   const response = await fetch(url) | ||||||
|  |   if (!response.ok) { | ||||||
|  |     throw new Error(`HTTP ${response.status}`) | ||||||
|  |   } | ||||||
|  |   return await response.json() | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | すべてのAPI呼び出しでエラーハンドリングを実装。 | ||||||
|  |  | ||||||
|  | ### 3. コンポーネント設計 | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // ❌ Bad: ビジネスロジックがコンポーネント内 | ||||||
|  | function MyComponent() { | ||||||
|  |   const [data, setData] = useState([]) | ||||||
|  |   useEffect(() => { | ||||||
|  |     fetch('/api/data').then(setData) | ||||||
|  |   }, []) | ||||||
|  |   return <div>{data.map(...)}</div> | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ✅ Good: Hooksでロジック分離 | ||||||
|  | function MyComponent() { | ||||||
|  |   const { data, loading, error } = useMyData() | ||||||
|  |   if (loading) return <Loading /> | ||||||
|  |   if (error) return <Error /> | ||||||
|  |   return <div>{data.map(...)}</div> | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## デバッグ手法 | ||||||
|  |  | ||||||
|  | ### 1. OAuth デバッグ | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // ブラウザの開発者ツールで確認 | ||||||
|  | localStorage.clear()          // セッションクリア | ||||||
|  | sessionStorage.clear()        // 一時データクリア | ||||||
|  |  | ||||||
|  | // IndexedDB確認(Application タブ) | ||||||
|  | // ATProtoの認証データが保存される | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 2. PDS判定デバッグ | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/utils/pds.js にログ追加 | ||||||
|  | console.log('Handle:', handle) | ||||||
|  | console.log('Is syu.is:', isSyuIsHandle(handle)) | ||||||
|  | console.log('API Config:', getApiConfig(pds)) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 3. レコードフィルタリングデバッグ | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/components/RecordTabs.jsx | ||||||
|  | console.log('Page Context:', pageContext) | ||||||
|  | console.log('All Records:', records.length) | ||||||
|  | console.log('Filtered Records:', filteredRecords.length) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## パフォーマンス最適化 | ||||||
|  |  | ||||||
|  | ### 1. 並列データ取得 | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/hooks/useAdminData.js | ||||||
|  | const [records, lang, comment] = await Promise.all([ | ||||||
|  |   collections.getBase(apiConfig.pds, did, env.collection), | ||||||
|  |   collections.getLang(apiConfig.pds, did, env.collection), | ||||||
|  |   collections.getComment(apiConfig.pds, did, env.collection) | ||||||
|  | ]) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 2. 不要な再レンダリング防止 | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // useMemo でフィルタリング結果をキャッシュ | ||||||
|  | const filteredRecords = useMemo(() =>  | ||||||
|  |   filterRecords(records),  | ||||||
|  |   [records, pageContext] | ||||||
|  | ) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## テスト戦略 | ||||||
|  |  | ||||||
|  | ### 1. 単体テスト推奨対象 | ||||||
|  |  | ||||||
|  | - `src/utils/pds.js` - PDS判定ロジック | ||||||
|  | - `src/config/env.js` - 環境変数パース | ||||||
|  | - フィルタリング関数 | ||||||
|  |  | ||||||
|  | ### 2. 統合テスト推奨対象 | ||||||
|  |  | ||||||
|  | - OAuth認証フロー | ||||||
|  | - API呼び出し | ||||||
|  | - レコード表示 | ||||||
|  |  | ||||||
|  | ## デプロイメント | ||||||
|  |  | ||||||
|  | ### 1. 必要ファイル | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | public/ | ||||||
|  | └── client-metadata.json    # OAuth設定ファイル | ||||||
|  |  | ||||||
|  | dist/                       # ビルド出力 | ||||||
|  | ├── index.html | ||||||
|  | └── assets/ | ||||||
|  |     ├── comment-atproto-[hash].js | ||||||
|  |     └── comment-atproto-[hash].css | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 2. デプロイ手順 | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # 1. 環境変数設定 | ||||||
|  | cp .env.example .env | ||||||
|  | # 2. 本番用設定を記入 | ||||||
|  | # 3. ビルド | ||||||
|  | npm run build | ||||||
|  | # 4. dist/ フォルダをデプロイ | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 3. 本番環境チェックリスト | ||||||
|  |  | ||||||
|  | - [ ] `.env` ファイルの本番設定 | ||||||
|  | - [ ] `client-metadata.json` の設置 | ||||||
|  | - [ ] HTTPS 必須(OAuth要件) | ||||||
|  | - [ ] CSP(Content Security Policy)設定 | ||||||
|  |  | ||||||
|  | ## よくある問題と解決法 | ||||||
|  |  | ||||||
|  | ### 1. "OAuth initialization failed" | ||||||
|  |  | ||||||
|  | **原因**: client-metadata.json が見つからない、または形式が正しくない | ||||||
|  |  | ||||||
|  | **解決法**:  | ||||||
|  | ```bash | ||||||
|  | # public/client-metadata.json の存在確認 | ||||||
|  | ls -la public/client-metadata.json | ||||||
|  |  | ||||||
|  | # 形式確認(JSON validation) | ||||||
|  | jq . public/client-metadata.json | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 2. "Failed to load admin data" | ||||||
|  |  | ||||||
|  | **原因**: 管理者アカウントのDID解決に失敗 | ||||||
|  |  | ||||||
|  | **解決法**: | ||||||
|  | ```bash | ||||||
|  | # 手動でDID解決確認 | ||||||
|  | curl "https://syu.is/xrpc/com.atproto.repo.describeRepo?repo=ai.syui.ai" | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 3. レコードが表示されない | ||||||
|  |  | ||||||
|  | **原因**: コレクション名の不一致、権限不足 | ||||||
|  |  | ||||||
|  | **解決法**: | ||||||
|  | ```bash | ||||||
|  | # コレクション確認 | ||||||
|  | curl "https://syu.is/xrpc/com.atproto.repo.listRecords?repo=did:plc:xxx&collection=ai.syui.log.chat.lang" | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 機能拡張ガイド | ||||||
|  |  | ||||||
|  | ### 1. 新しいコレクション追加 | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/api/atproto.js に追加 | ||||||
|  | export const collections = { | ||||||
|  |   // 既存... | ||||||
|  |   async getNewCollection(pds, repo, collection, limit = 10) { | ||||||
|  |     return await atproto.getRecords(pds, repo, `${collection}.new`, limit) | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 2. 新しいPDS対応 | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/utils/pds.js を拡張 | ||||||
|  | export function getApiConfig(pds) { | ||||||
|  |   if (pds.includes('syu.is')) { | ||||||
|  |     // 既存の syu.is 設定 | ||||||
|  |   } else if (pds.includes('newpds.com')) { | ||||||
|  |     return { | ||||||
|  |       pds: `https://newpds.com`, | ||||||
|  |       bsky: `https://bsky.newpds.com`, | ||||||
|  |       plc: `https://plc.newpds.com`, | ||||||
|  |       web: `https://web.newpds.com` | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   // デフォルト設定 | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 3. リアルタイム更新追加 | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // src/hooks/useRealtimeUpdates.js | ||||||
|  | export function useRealtimeUpdates(collection) { | ||||||
|  |   useEffect(() => { | ||||||
|  |     const ws = new WebSocket('wss://jetstream2.us-east.bsky.network/subscribe') | ||||||
|  |     ws.onmessage = (event) => { | ||||||
|  |       const data = JSON.parse(event.data) | ||||||
|  |       if (data.collection === collection) { | ||||||
|  |         // 新しいレコードを追加 | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return () => ws.close() | ||||||
|  |   }, [collection]) | ||||||
|  | } | ||||||
|  | ``` | ||||||
							
								
								
									
										222
									
								
								oauth_new/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										222
									
								
								oauth_new/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,222 @@ | |||||||
|  | # ATProto OAuth Comment System | ||||||
|  |  | ||||||
|  | ATProtocol(Bluesky)のOAuth認証を使用したコメントシステムです。 | ||||||
|  |  | ||||||
|  | ## プロジェクト概要 | ||||||
|  |  | ||||||
|  | このプロジェクトは、ATProtocolネットワーク上のコメントとlangレコードを表示するWebアプリケーションです。 | ||||||
|  | - 標準的なOAuth認証画面を使用 | ||||||
|  | - タブ切り替えでレコード表示 | ||||||
|  | - ページコンテキストに応じたフィルタリング | ||||||
|  |  | ||||||
|  | ## ファイル構成 | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | src/ | ||||||
|  | ├── config/ | ||||||
|  | │   └── env.js              # 環境変数の一元管理 | ||||||
|  | ├── utils/ | ||||||
|  | │   └── pds.js              # PDS判定・API設定ユーティリティ | ||||||
|  | ├── api/ | ||||||
|  | │   └── atproto.js          # ATProto API クライアント | ||||||
|  | ├── hooks/ | ||||||
|  | │   ├── useAuth.js          # OAuth認証フック | ||||||
|  | │   ├── useAdminData.js     # 管理者データ取得フック | ||||||
|  | │   └── usePageContext.js   # ページ判定フック | ||||||
|  | ├── services/ | ||||||
|  | │   └── oauth.js            # OAuth認証サービス | ||||||
|  | ├── components/ | ||||||
|  | │   ├── AuthButton.jsx      # ログイン/ログアウトボタン | ||||||
|  | │   ├── RecordTabs.jsx      # Lang/Commentタブ切り替え | ||||||
|  | │   ├── RecordList.jsx      # レコード表示リスト | ||||||
|  | │   ├── UserLookup.jsx      # ユーザー検索(未使用) | ||||||
|  | │   └── OAuthCallback.jsx   # OAuth コールバック処理 | ||||||
|  | └── App.jsx                 # メインアプリケーション | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 環境設定 | ||||||
|  |  | ||||||
|  | ### .env ファイル | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | VITE_ADMIN=ai.syui.ai                                    # 管理者ハンドル | ||||||
|  | VITE_PDS=syu.is                                         # デフォルトPDS | ||||||
|  | VITE_HANDLE_LIST=["ai.syui.ai", "syui.syui.ai", "ai.ai"] # syu.is系ハンドルリスト | ||||||
|  | VITE_COLLECTION=ai.syui.log                             # ベースコレクション | ||||||
|  | VITE_OAUTH_CLIENT_ID=https://syui.ai/client-metadata.json # OAuth クライアントID | ||||||
|  | VITE_OAUTH_REDIRECT_URI=https://syui.ai/oauth/callback   # OAuth リダイレクトURI | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 必要な依存関係 | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |   "dependencies": { | ||||||
|  |     "@atproto/api": "^0.15.12", | ||||||
|  |     "@atproto/oauth-client-browser": "^0.3.19", | ||||||
|  |     "react": "^18.2.0", | ||||||
|  |     "react-dom": "^18.2.0" | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 主要機能 | ||||||
|  |  | ||||||
|  | ### 1. OAuth認証システム | ||||||
|  |  | ||||||
|  | **実装場所**: `src/services/oauth.js` | ||||||
|  |  | ||||||
|  | - `@atproto/oauth-client-browser`を使用した標準OAuth実装 | ||||||
|  | - bsky.social と syu.is 両方のPDSに対応 | ||||||
|  | - セッション自動復元機能 | ||||||
|  |  | ||||||
|  | **重要**: ATProtoのセッション管理は複雑なため、公式ライブラリの使用が必須です。 | ||||||
|  |  | ||||||
|  | ### 2. PDS判定システム | ||||||
|  |  | ||||||
|  | **実装場所**: `src/utils/pds.js` | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // ハンドル判定ロジック | ||||||
|  | isSyuIsHandle(handle) → boolean | ||||||
|  | // PDS設定取得 | ||||||
|  | getApiConfig(pds) → { pds, bsky, plc, web } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | 環境変数`VITE_HANDLE_LIST`と`VITE_PDS`を基に自動判定します。 | ||||||
|  |  | ||||||
|  | ### 3. コレクション取得システム | ||||||
|  |  | ||||||
|  | **実装場所**: `src/api/atproto.js` | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // 基本コレクション | ||||||
|  | collections.getBase(pds, repo, collection) | ||||||
|  | // lang コレクション(翻訳系) | ||||||
|  | collections.getLang(pds, repo, collection) // → {collection}.chat.lang | ||||||
|  | // comment コレクション(コメント系) | ||||||
|  | collections.getComment(pds, repo, collection) // → {collection}.chat.comment | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### 4. ページコンテキスト判定 | ||||||
|  |  | ||||||
|  | **実装場所**: `src/hooks/usePageContext.js` | ||||||
|  |  | ||||||
|  | ```javascript | ||||||
|  | // URL解析結果 | ||||||
|  | { | ||||||
|  |   isTopPage: boolean,     // トップページかどうか | ||||||
|  |   rkey: string | null,    // 個別ページのrkey(/posts/xxx → xxx) | ||||||
|  |   url: string             // 現在のURL | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 表示ロジック | ||||||
|  |  | ||||||
|  | ### フィルタリング | ||||||
|  |  | ||||||
|  | 1. **トップページ**: 最新3件を表示 | ||||||
|  | 2. **個別ページ**: `record.value.post.url`の rkey が現在ページと一致するもののみ表示 | ||||||
|  |  | ||||||
|  | ### タブ切り替え | ||||||
|  |  | ||||||
|  | - Lang Records: `{collection}.chat.lang` | ||||||
|  | - Comment Records: `{collection}.chat.comment` | ||||||
|  |  | ||||||
|  | ## 開発・デバッグ | ||||||
|  |  | ||||||
|  | ### 起動コマンド | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | npm install | ||||||
|  | npm run dev    # 開発サーバー | ||||||
|  | npm run build  # プロダクションビルド | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### OAuth デバッグ | ||||||
|  |  | ||||||
|  | 1. **ローカル開発**: 自動的にloopback clientが使用される | ||||||
|  | 2. **本番環境**: `client-metadata.json`が必要 | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | // public/client-metadata.json | ||||||
|  | { | ||||||
|  |   "client_id": "https://syui.ai/client-metadata.json", | ||||||
|  |   "client_name": "ATProto Comment System", | ||||||
|  |   "redirect_uris": ["https://syui.ai/oauth/callback"], | ||||||
|  |   "scope": "atproto", | ||||||
|  |   "grant_types": ["authorization_code", "refresh_token"], | ||||||
|  |   "response_types": ["code"], | ||||||
|  |   "token_endpoint_auth_method": "none", | ||||||
|  |   "application_type": "web", | ||||||
|  |   "dpop_bound_access_tokens": true | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### よくある問題 | ||||||
|  |  | ||||||
|  | 1. **セッションが保存されない** | ||||||
|  |    - `@atproto/oauth-client-browser`のバージョン確認 | ||||||
|  |    - IndexedDBの確認(ブラウザの開発者ツール) | ||||||
|  |  | ||||||
|  | 2. **PDS判定が正しく動作しない** | ||||||
|  |    - `VITE_HANDLE_LIST`の JSON 形式を確認 | ||||||
|  |    - 環境変数の読み込み確認 | ||||||
|  |  | ||||||
|  | 3. **レコードが表示されない** | ||||||
|  |    - 管理者アカウントの DID 解決確認 | ||||||
|  |    - コレクション名の確認(`{base}.chat.lang`, `{base}.chat.comment`) | ||||||
|  |  | ||||||
|  | ## API エンドポイント | ||||||
|  |  | ||||||
|  | ### 使用しているATProto API | ||||||
|  |  | ||||||
|  | 1. **com.atproto.repo.describeRepo** | ||||||
|  |    - ハンドル → DID, PDS解決 | ||||||
|  |  | ||||||
|  | 2. **app.bsky.actor.getProfile** | ||||||
|  |    - プロフィール情報取得 | ||||||
|  |  | ||||||
|  | 3. **com.atproto.repo.listRecords** | ||||||
|  |    - コレクションレコード取得 | ||||||
|  |  | ||||||
|  | ## セキュリティ | ||||||
|  |  | ||||||
|  | - OAuth 2.1 + PKCE による認証 | ||||||
|  | - DPoP (Demonstration of Proof of Possession) 対応 | ||||||
|  | - セッション情報はブラウザのIndexedDBに暗号化保存 | ||||||
|  |  | ||||||
|  | ## 今後の拡張可能性 | ||||||
|  |  | ||||||
|  | 1. **コメント投稿機能** | ||||||
|  |    - 認証済みユーザーによるコメント作成 | ||||||
|  |    - `com.atproto.repo.putRecord` API使用 | ||||||
|  |  | ||||||
|  | 2. **リアルタイム更新** | ||||||
|  |    - Jetstream WebSocket 接続 | ||||||
|  |    - 新しいレコードの自動表示 | ||||||
|  |  | ||||||
|  | 3. **マルチPDS対応** | ||||||
|  |    - より多くのPDSへの対応 | ||||||
|  |    - 動的PDS判定の改善 | ||||||
|  |  | ||||||
|  | ## トラブルシューティング | ||||||
|  |  | ||||||
|  | ### ログ確認 | ||||||
|  | ブラウザの開発者ツールでコンソールログを確認してください。主要なエラーは以下の通りです: | ||||||
|  |  | ||||||
|  | - `OAuth initialization failed`: OAuth設定の問題 | ||||||
|  | - `Failed to load admin data`: API アクセスエラー | ||||||
|  | - `Auth check failed`: セッション復元エラー | ||||||
|  |  | ||||||
|  | ### 環境変数確認 | ||||||
|  | ```javascript | ||||||
|  | // 開発者ツールのコンソールで確認 | ||||||
|  | console.log(import.meta.env) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## 参考資料 | ||||||
|  |  | ||||||
|  | - [ATProto OAuth Guide](https://github.com/bluesky-social/atproto/blob/main/packages/api/OAUTH.md) | ||||||
|  | - [BrowserOAuthClient Documentation](https://github.com/bluesky-social/atproto/tree/main/packages/oauth-client-browser) | ||||||
|  | - [ATProto API Reference](https://docs.bsky.app/docs/advanced-guides/atproto-api) | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user