5 Commits

Author SHA1 Message Date
2838e494d6 Update version to 0.1.3 and simplify version output
- Update Cargo.toml version to 0.1.3
- Custom version flag implementation to output version only
- Update GitHub Actions to handle simplified version output
- Update Gitea Actions (action.yml) for version handling
- Now `ailog -V` or `ailog --version` outputs just "0.1.3"

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-06-14 17:12:04 +09:00
5838adf5a6 Fix code block file path display for mobile
- Move file path from right-top overlay to top header bar
- Use block display instead of absolute positioning
- Add dedicated styling for filename header bar
- Adjust code padding when filename is present
- Optimize mobile responsiveness to prevent overlap

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-06-14 17:03:31 +09:00
f52249c292 Fix GitHub Actions permissions for release creation
- Add contents: write permission for private repository
- Remove Windows zip file references
- Simplify artifact upload paths
- Fix release workflow permissions for private repos

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-06-14 16:53:06 +09:00
aa70183d75 Fix cross-compilation issues and optimize release workflow
- Fix ARM64 cross-compilation with proper binutils
- Use target-specific strip commands
- Remove Windows/musl builds to reduce complexity
- Add 60-minute timeout for builds
- Optimize to 4 core platforms: Linux x86_64/ARM64, macOS Intel/ARM

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-06-14 16:42:28 +09:00
4ad1d3edf6 Fix cross-compilation issues and private repo access
- Replace native-tls with rustls-tls for cross-platform compatibility
- Add vendored OpenSSL/libgit2 features for static linking
- Add connect feature to tokio-tungstenite
- Add GITHUB_TOKEN authentication for private repo access
- Add smart binary caching with version checking workflow

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-06-14 16:26:36 +09:00
7 changed files with 160 additions and 69 deletions

View File

@ -42,7 +42,10 @@
"WebFetch(domain:syui.ai)", "WebFetch(domain:syui.ai)",
"Bash(rustup target list:*)", "Bash(rustup target list:*)",
"Bash(rustup target:*)", "Bash(rustup target:*)",
"Bash(git add:*)" "Bash(git add:*)",
"Bash(git commit:*)",
"Bash(git push:*)",
"Bash(git tag:*)"
], ],
"deny": [] "deny": []
} }

View File

@ -15,18 +15,45 @@ jobs:
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
- name: Get latest release - name: Cache ailog binary
id: latest_release uses: actions/cache@v4
run: | with:
LATEST_TAG=$(curl -s https://api.github.com/repos/${{ github.repository }}/releases/latest | jq -r .tag_name) path: ./bin
echo "tag=$LATEST_TAG" >> $GITHUB_OUTPUT key: ailog-bin-${{ runner.os }}
restore-keys: |
ailog-bin-${{ runner.os }}
- name: Download pre-built binary from release - name: Check and update ailog binary
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
run: | run: |
curl -sL https://github.com/${{ github.repository }}/releases/download/${{ steps.latest_release.outputs.tag }}/ailog-linux-x86_64.tar.gz | tar -xzf - # Get latest release version
chmod +x ailog LATEST_VERSION=$(curl -s -H "Authorization: Bearer $GITHUB_TOKEN" \
https://api.github.com/repos/${{ github.repository }}/releases/latest | jq -r .tag_name)
echo "Latest version: $LATEST_VERSION"
# Check current binary version if exists
mkdir -p ./bin mkdir -p ./bin
mv ailog ./bin/ if [ -f "./bin/ailog" ]; then
CURRENT_VERSION=$(./bin/ailog --version 2>/dev/null || echo "unknown")
echo "Current version: $CURRENT_VERSION"
else
CURRENT_VERSION="none"
echo "No binary found"
fi
# Download if version is different or binary doesn't exist
if [ "$CURRENT_VERSION" != "${LATEST_VERSION#v}" ]; then
echo "Downloading ailog $LATEST_VERSION..."
curl -sL -H "Authorization: Bearer $GITHUB_TOKEN" \
https://github.com/${{ github.repository }}/releases/download/$LATEST_VERSION/ailog-linux-x86_64.tar.gz | tar -xzf -
mv ailog ./bin/ailog
chmod +x ./bin/ailog
echo "Updated to version: $(./bin/ailog --version 2>/dev/null)"
else
echo "Binary is up to date"
chmod +x ./bin/ailog
fi
- name: Setup Hugo - name: Setup Hugo
uses: peaceiris/actions-hugo@v3 uses: peaceiris/actions-hugo@v3

View File

@ -11,13 +11,20 @@ on:
required: true required: true
default: 'v0.1.0' default: 'v0.1.0'
permissions:
contents: write
actions: read
env: env:
CARGO_TERM_COLOR: always CARGO_TERM_COLOR: always
OPENSSL_STATIC: true
OPENSSL_VENDOR: true
jobs: jobs:
build: build:
name: Build ${{ matrix.target }} name: Build ${{ matrix.target }}
runs-on: ${{ matrix.os }} runs-on: ${{ matrix.os }}
timeout-minutes: 60
strategy: strategy:
matrix: matrix:
include: include:
@ -25,10 +32,6 @@ jobs:
os: ubuntu-latest os: ubuntu-latest
artifact_name: ailog artifact_name: ailog
asset_name: ailog-linux-x86_64 asset_name: ailog-linux-x86_64
- target: x86_64-unknown-linux-musl
os: ubuntu-latest
artifact_name: ailog
asset_name: ailog-linux-x86_64-musl
- target: aarch64-unknown-linux-gnu - target: aarch64-unknown-linux-gnu
os: ubuntu-latest os: ubuntu-latest
artifact_name: ailog artifact_name: ailog
@ -41,10 +44,6 @@ jobs:
os: macos-latest os: macos-latest
artifact_name: ailog artifact_name: ailog
asset_name: ailog-macos-aarch64 asset_name: ailog-macos-aarch64
- target: x86_64-pc-windows-msvc
os: windows-latest
artifact_name: ailog.exe
asset_name: ailog-windows-x86_64.exe
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
@ -55,16 +54,10 @@ jobs:
targets: ${{ matrix.target }} targets: ${{ matrix.target }}
- name: Install cross-compilation tools (Linux) - name: Install cross-compilation tools (Linux)
if: matrix.os == 'ubuntu-latest' if: matrix.os == 'ubuntu-latest' && matrix.target == 'aarch64-unknown-linux-gnu'
run: | run: |
sudo apt-get update sudo apt-get update
sudo apt-get install -y gcc-multilib sudo apt-get install -y gcc-aarch64-linux-gnu binutils-aarch64-linux-gnu
if [[ "${{ matrix.target }}" == "aarch64-unknown-linux-gnu" ]]; then
sudo apt-get install -y gcc-aarch64-linux-gnu
fi
if [[ "${{ matrix.target }}" == "x86_64-unknown-linux-musl" ]]; then
sudo apt-get install -y musl-tools
fi
- name: Configure cross-compilation (Linux ARM64) - name: Configure cross-compilation (Linux ARM64)
if: matrix.target == 'aarch64-unknown-linux-gnu' if: matrix.target == 'aarch64-unknown-linux-gnu'
@ -93,11 +86,17 @@ jobs:
shell: bash shell: bash
run: | run: |
cd target/${{ matrix.target }}/release cd target/${{ matrix.target }}/release
if [[ "${{ matrix.os }}" == "windows-latest" ]]; then
strip ${{ matrix.artifact_name }} || true # 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 else
strip ${{ matrix.artifact_name }} strip ${{ matrix.artifact_name }} || echo "Strip failed, continuing..."
fi fi
# Create archive
if [[ "${{ matrix.target }}" == *"windows"* ]]; then if [[ "${{ matrix.target }}" == *"windows"* ]]; then
7z a ../../../${{ matrix.asset_name }}.zip ${{ matrix.artifact_name }} 7z a ../../../${{ matrix.asset_name }}.zip ${{ matrix.artifact_name }}
else else
@ -108,14 +107,15 @@ jobs:
uses: actions/upload-artifact@v4 uses: actions/upload-artifact@v4
with: with:
name: ${{ matrix.asset_name }} name: ${{ matrix.asset_name }}
path: | path: ${{ matrix.asset_name }}.tar.gz
${{ matrix.asset_name }}.tar.gz
${{ matrix.asset_name }}.zip
release: release:
name: Create Release name: Create Release
needs: build needs: build
runs-on: ubuntu-latest runs-on: ubuntu-latest
permissions:
contents: write
actions: read
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
@ -135,9 +135,8 @@ jobs:
echo "- AI comment system" >> release_notes.md echo "- AI comment system" >> release_notes.md
echo "" >> release_notes.md echo "" >> release_notes.md
echo "### Platforms" >> release_notes.md echo "### Platforms" >> release_notes.md
echo "- Linux (x86_64, aarch64, musl)" >> release_notes.md echo "- Linux (x86_64, aarch64)" >> release_notes.md
echo "- macOS (Intel, Apple Silicon)" >> release_notes.md echo "- macOS (Intel, Apple Silicon)" >> release_notes.md
echo "- Windows (x86_64)" >> release_notes.md
echo "" >> release_notes.md echo "" >> release_notes.md
echo "### Installation" >> release_notes.md echo "### Installation" >> release_notes.md
echo "\`\`\`bash" >> release_notes.md echo "\`\`\`bash" >> release_notes.md
@ -146,8 +145,6 @@ jobs:
echo "chmod +x ailog" >> release_notes.md echo "chmod +x ailog" >> release_notes.md
echo "sudo mv ailog /usr/local/bin/" >> release_notes.md echo "sudo mv ailog /usr/local/bin/" >> release_notes.md
echo "" >> release_notes.md echo "" >> release_notes.md
echo "# Windows" >> release_notes.md
echo "# Extract ailog-windows-x86_64.zip and add to PATH" >> release_notes.md
echo "\`\`\`" >> release_notes.md echo "\`\`\`" >> release_notes.md
- name: Get tag name - name: Get tag name
@ -167,8 +164,6 @@ jobs:
body_path: release_notes.md body_path: release_notes.md
draft: false draft: false
prerelease: ${{ contains(steps.tag_name.outputs.tag, 'alpha') || contains(steps.tag_name.outputs.tag, 'beta') || contains(steps.tag_name.outputs.tag, 'rc') }} prerelease: ${{ contains(steps.tag_name.outputs.tag, 'alpha') || contains(steps.tag_name.outputs.tag, 'beta') || contains(steps.tag_name.outputs.tag, 'rc') }}
files: | files: artifacts/*/ailog-*.tar.gz
artifacts/*/ailog-*.tar.gz
artifacts/*/ailog-*.zip
env: env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

View File

@ -1,6 +1,6 @@
[package] [package]
name = "ailog" name = "ailog"
version = "0.1.0" version = "0.1.3"
edition = "2021" edition = "2021"
authors = ["syui"] authors = ["syui"]
description = "A static blog generator with AI features" description = "A static blog generator with AI features"
@ -26,7 +26,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"
@ -43,12 +43,12 @@ cookie = "0.18"
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 # ATProto and stream monitoring dependencies
tokio-tungstenite = { version = "0.21", features = ["native-tls"] } tokio-tungstenite = { version = "0.21", features = ["rustls-tls-webpki-roots", "connect"], default-features = false }
futures-util = "0.3" futures-util = "0.3"
tungstenite = { version = "0.21", features = ["native-tls"] } tungstenite = { version = "0.21", features = ["rustls-tls-webpki-roots"], default-features = false }
[dev-dependencies] [dev-dependencies]
tempfile = "3.14" tempfile = "3.14"

View File

@ -47,17 +47,57 @@ outputs:
runs: runs:
using: 'composite' using: 'composite'
steps: steps:
- name: Setup Rust - name: Cache ailog binary
uses: actions-rs/toolchain@v1 uses: actions/cache@v4
with: with:
toolchain: stable path: ./bin
override: true key: ailog-bin-${{ runner.os }}
restore-keys: |
ailog-bin-${{ runner.os }}
- name: Install ailog - name: Check and update ailog binary
shell: bash shell: bash
run: | run: |
if [ ! -f "./target/release/ailog" ]; then # Get latest release version (for Gitea, adjust API endpoint if needed)
if command -v curl >/dev/null 2>&1; then
LATEST_VERSION=$(curl -s https://api.github.com/repos/syui/ailog/releases/latest | jq -r .tag_name 2>/dev/null || echo "v0.1.1")
else
LATEST_VERSION="v0.1.1" # fallback version
fi
echo "Target version: $LATEST_VERSION"
# Check current binary version if exists
mkdir -p ./bin
if [ -f "./bin/ailog" ]; then
CURRENT_VERSION=$(./bin/ailog --version 2>/dev/null || echo "unknown")
echo "Current version: $CURRENT_VERSION"
else
CURRENT_VERSION="none"
echo "No binary found"
fi
# Download if version is different or binary doesn't exist
if [ "$CURRENT_VERSION" != "${LATEST_VERSION#v}" ]; then
echo "Downloading ailog $LATEST_VERSION..."
# Try GitHub first, then fallback to local build
if curl -sL https://github.com/syui/ailog/releases/download/$LATEST_VERSION/ailog-linux-x86_64.tar.gz | tar -xzf - 2>/dev/null; then
mv ailog ./bin/ailog
chmod +x ./bin/ailog
echo "Downloaded binary: $(./bin/ailog --version 2>/dev/null)"
else
echo "Download failed, building from source..."
if command -v cargo >/dev/null 2>&1; then
cargo build --release cargo build --release
cp ./target/release/ailog ./bin/ailog
echo "Built from source: $(./bin/ailog --version 2>/dev/null)"
else
echo "Error: Neither download nor cargo build available"
exit 1
fi
fi
else
echo "Binary is up to date"
chmod +x ./bin/ailog
fi fi
- name: Setup Node.js for OAuth app - name: Setup Node.js for OAuth app
@ -83,8 +123,8 @@ runs:
run: | run: |
start_time=$(date +%s) start_time=$(date +%s)
./target/release/ailog generate \ ./bin/ailog build \
--input ${{ inputs.content-dir }} \ --content ${{ inputs.content-dir }} \
--output ${{ inputs.output-dir }} \ --output ${{ inputs.output-dir }} \
--templates ${{ inputs.template-dir }} \ --templates ${{ inputs.template-dir }} \
--static ${{ inputs.static-dir }} \ --static ${{ inputs.static-dir }} \

View File

@ -517,25 +517,21 @@ a.view-markdown:any-link {
margin: 16px 0; margin: 16px 0;
font-size: 14px; font-size: 14px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.3); box-shadow: 0 2px 8px rgba(0, 0, 0, 0.3);
position: relative;
} }
/* File name display for code blocks */ /* File name display for code blocks - top bar style */
.article-body pre[data-filename]::before { .article-body pre[data-filename]::before {
content: attr(data-filename); content: attr(data-filename);
position: absolute; display: block;
top: 0;
right: 0;
background: #2D2D30; background: #2D2D30;
color: #CCCCCC; color: #AE81FF;
padding: 4px 12px; padding: 8px 16px;
font-size: 12px; font-size: 12px;
font-family: 'SF Mono', 'Monaco', 'Cascadia Code', 'Roboto Mono', monospace; font-family: 'SF Mono', 'Monaco', 'Cascadia Code', 'Roboto Mono', monospace;
border-bottom-left-radius: 4px; border-bottom: 1px solid #3E3D32;
border: 1px solid #3E3D32; margin: 0;
border-top: none; width: 100%;
border-right: none; box-sizing: border-box;
z-index: 1;
} }
.article-body pre code { .article-body pre code {
@ -548,6 +544,11 @@ a.view-markdown:any-link {
line-height: 1.4; line-height: 1.4;
} }
/* Adjust padding when filename is present */
.article-body pre[data-filename] code {
padding: 16px;
}
/* Inline code (not in pre blocks) */ /* Inline code (not in pre blocks) */
.article-body code { .article-body code {
background: var(--light-white); background: var(--light-white);
@ -853,6 +854,16 @@ a.view-markdown:any-link {
white-space: pre-wrap; 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 { .article-body code {
word-break: break-all; word-break: break-all;
} }

View File

@ -18,10 +18,14 @@ mod mcp;
#[derive(Parser)] #[derive(Parser)]
#[command(name = "ailog")] #[command(name = "ailog")]
#[command(about = "A static blog generator with AI features")] #[command(about = "A static blog generator with AI features")]
#[command(version)] #[command(disable_version_flag = true)]
struct Cli { struct Cli {
/// Print version information
#[arg(short = 'V', long = "version")]
version: bool,
#[command(subcommand)] #[command(subcommand)]
command: Commands, command: Option<Commands>,
} }
#[derive(Subcommand)] #[derive(Subcommand)]
@ -136,7 +140,18 @@ enum OauthCommands {
async fn main() -> Result<()> { async fn main() -> Result<()> {
let cli = Cli::parse(); let cli = Cli::parse();
match cli.command { // Handle version flag
if cli.version {
println!("{}", env!("CARGO_PKG_VERSION"));
return Ok(());
}
// Require subcommand if no version flag
let command = cli.command.ok_or_else(|| {
anyhow::anyhow!("No subcommand provided. Use --help for usage information.")
})?;
match command {
Commands::Init { path } => { Commands::Init { path } => {
commands::init::execute(path).await?; commands::init::execute(path).await?;
} }