mirror of
https://mirror.suhoan.cn/https://github.com/EasyTier/EasyTier.git
synced 2025-12-13 21:27:25 +08:00
add binary file easytier-web-embed (#718)
* embed web dashboard into easytier-web * add binary file easytier-web-embed
This commit is contained in:
79
.github/workflows/core.yml
vendored
79
.github/workflows/core.yml
vendored
@@ -31,6 +31,47 @@ jobs:
|
|||||||
skip_after_successful_duplicate: 'true'
|
skip_after_successful_duplicate: 'true'
|
||||||
cancel_others: 'true'
|
cancel_others: 'true'
|
||||||
paths: '["Cargo.toml", "Cargo.lock", "easytier/**", ".github/workflows/core.yml", ".github/workflows/install_rust.sh"]'
|
paths: '["Cargo.toml", "Cargo.lock", "easytier/**", ".github/workflows/core.yml", ".github/workflows/install_rust.sh"]'
|
||||||
|
build_web:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: pre_job
|
||||||
|
if: needs.pre_job.outputs.should_skip != 'true'
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 21
|
||||||
|
|
||||||
|
- name: Install pnpm
|
||||||
|
uses: pnpm/action-setup@v3
|
||||||
|
with:
|
||||||
|
version: 9
|
||||||
|
run_install: false
|
||||||
|
|
||||||
|
- name: Get pnpm store directory
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
echo "STORE_PATH=$(pnpm store path --silent)" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Setup pnpm cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ env.STORE_PATH }}
|
||||||
|
key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-pnpm-store-
|
||||||
|
|
||||||
|
- name: Install frontend dependencies
|
||||||
|
run: |
|
||||||
|
pnpm -r install
|
||||||
|
pnpm -r --filter "./easytier-web/*" build
|
||||||
|
|
||||||
|
- name: Archive artifact
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: easytier-web-dashboard
|
||||||
|
path: |
|
||||||
|
easytier-web/frontend/dist/*
|
||||||
build:
|
build:
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
@@ -87,7 +128,9 @@ jobs:
|
|||||||
TARGET: ${{ matrix.TARGET }}
|
TARGET: ${{ matrix.TARGET }}
|
||||||
OS: ${{ matrix.OS }}
|
OS: ${{ matrix.OS }}
|
||||||
OSS_BUCKET: ${{ secrets.ALIYUN_OSS_BUCKET }}
|
OSS_BUCKET: ${{ secrets.ALIYUN_OSS_BUCKET }}
|
||||||
needs: pre_job
|
needs:
|
||||||
|
- pre_job
|
||||||
|
- build_web
|
||||||
if: needs.pre_job.outputs.should_skip != 'true'
|
if: needs.pre_job.outputs.should_skip != 'true'
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
@@ -96,6 +139,12 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
echo "GIT_DESC=$(git log -1 --format=%cd.%h --date=format:%Y-%m-%d_%H:%M:%S)" >> $GITHUB_ENV
|
echo "GIT_DESC=$(git log -1 --format=%cd.%h --date=format:%Y-%m-%d_%H:%M:%S)" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Download web artifact
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
name: easytier-web-dashboard
|
||||||
|
path: easytier-web/frontend/dist/
|
||||||
|
|
||||||
- name: Cargo cache
|
- name: Cargo cache
|
||||||
if: ${{ ! endsWith(matrix.TARGET, 'freebsd') }}
|
if: ${{ ! endsWith(matrix.TARGET, 'freebsd') }}
|
||||||
uses: actions/cache@v4
|
uses: actions/cache@v4
|
||||||
@@ -121,23 +170,27 @@ jobs:
|
|||||||
if [[ $OS =~ ^ubuntu.*$ && $TARGET =~ ^mips.*$ ]]; then
|
if [[ $OS =~ ^ubuntu.*$ && $TARGET =~ ^mips.*$ ]]; then
|
||||||
cargo +nightly build -r --verbose --target $TARGET -Z build-std=std,panic_abort --no-default-features --features mips --package=easytier
|
cargo +nightly build -r --verbose --target $TARGET -Z build-std=std,panic_abort --no-default-features --features mips --package=easytier
|
||||||
else
|
else
|
||||||
|
if [[ $OS =~ ^windows.*$ ]]; then
|
||||||
|
SUFFIX=.exe
|
||||||
|
fi
|
||||||
|
cargo build --release --verbose --target $TARGET --package=easytier-web --features=embed
|
||||||
|
mv ./target/$TARGET/release/easytier-web"$SUFFIX" ./target/$TARGET/release/easytier-web-embed"$SUFFIX"
|
||||||
cargo build --release --verbose --target $TARGET
|
cargo build --release --verbose --target $TARGET
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Copied and slightly modified from @lmq8267 (https://github.com/lmq8267)
|
# Copied and slightly modified from @lmq8267 (https://github.com/lmq8267)
|
||||||
- name: Build Core & Cli (X86_64 FreeBSD)
|
- name: Build Core & Cli (X86_64 FreeBSD)
|
||||||
uses: cross-platform-actions/action@v0.23.0
|
uses: vmactions/freebsd-vm@v1
|
||||||
if: ${{ endsWith(matrix.TARGET, 'freebsd') }}
|
if: ${{ endsWith(matrix.TARGET, 'freebsd') }}
|
||||||
env:
|
env:
|
||||||
TARGET: ${{ matrix.TARGET }}
|
TARGET: ${{ matrix.TARGET }}
|
||||||
with:
|
with:
|
||||||
operating_system: freebsd
|
envs: TARGET
|
||||||
environment_variables: TARGET
|
release: ${{ matrix.BSD_VERSION }}
|
||||||
architecture: x86-64
|
arch: x86_64
|
||||||
version: ${{ matrix.BSD_VERSION }}
|
usesh: true
|
||||||
shell: bash
|
mem: 6144
|
||||||
memory: 5G
|
cpu: 4
|
||||||
cpu_count: 4
|
|
||||||
run: |
|
run: |
|
||||||
uname -a
|
uname -a
|
||||||
echo $SHELL
|
echo $SHELL
|
||||||
@@ -146,9 +199,9 @@ jobs:
|
|||||||
whoami
|
whoami
|
||||||
env | sort
|
env | sort
|
||||||
|
|
||||||
sudo pkg install -y git protobuf llvm-devel
|
pkg install -y git protobuf llvm-devel sudo curl
|
||||||
curl --proto 'https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y
|
curl --proto 'https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y
|
||||||
source $HOME/.cargo/env
|
. $HOME/.cargo/env
|
||||||
|
|
||||||
rustup set auto-self-update disable
|
rustup set auto-self-update disable
|
||||||
|
|
||||||
@@ -159,6 +212,8 @@ jobs:
|
|||||||
export CXX=clang++
|
export CXX=clang++
|
||||||
export CARGO_TERM_COLOR=always
|
export CARGO_TERM_COLOR=always
|
||||||
|
|
||||||
|
cargo build --release --verbose --target $TARGET --package=easytier-web --features=embed
|
||||||
|
mv ./target/$TARGET/release/easytier-web ./target/$TARGET/release/easytier-web-embed
|
||||||
cargo build --release --verbose --target $TARGET
|
cargo build --release --verbose --target $TARGET
|
||||||
|
|
||||||
- name: Install UPX
|
- name: Install UPX
|
||||||
@@ -196,6 +251,7 @@ jobs:
|
|||||||
mv ./target/$TARGET/release/easytier-cli"$SUFFIX" ./artifacts/objects/
|
mv ./target/$TARGET/release/easytier-cli"$SUFFIX" ./artifacts/objects/
|
||||||
if [[ ! $TARGET =~ ^mips.*$ ]]; then
|
if [[ ! $TARGET =~ ^mips.*$ ]]; then
|
||||||
mv ./target/$TARGET/release/easytier-web"$SUFFIX" ./artifacts/objects/
|
mv ./target/$TARGET/release/easytier-web"$SUFFIX" ./artifacts/objects/
|
||||||
|
mv ./target/$TARGET/release/easytier-web-embed"$SUFFIX" ./artifacts/objects/
|
||||||
fi
|
fi
|
||||||
|
|
||||||
mv ./artifacts/objects/* ./artifacts/
|
mv ./artifacts/objects/* ./artifacts/
|
||||||
@@ -213,6 +269,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs:
|
needs:
|
||||||
- pre_job
|
- pre_job
|
||||||
|
- build_web
|
||||||
- build
|
- build
|
||||||
steps:
|
steps:
|
||||||
- name: Mark result as failed
|
- name: Mark result as failed
|
||||||
|
|||||||
28
.github/workflows/test.yml
vendored
28
.github/workflows/test.yml
vendored
@@ -52,6 +52,34 @@ jobs:
|
|||||||
sudo sysctl net.ipv6.conf.lo.disable_ipv6=0
|
sudo sysctl net.ipv6.conf.lo.disable_ipv6=0
|
||||||
sudo ip addr add 2001:db8::2/64 dev lo
|
sudo ip addr add 2001:db8::2/64 dev lo
|
||||||
|
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 21
|
||||||
|
|
||||||
|
- name: Install pnpm
|
||||||
|
uses: pnpm/action-setup@v3
|
||||||
|
with:
|
||||||
|
version: 9
|
||||||
|
run_install: false
|
||||||
|
|
||||||
|
- name: Get pnpm store directory
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
echo "STORE_PATH=$(pnpm store path --silent)" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Setup pnpm cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ env.STORE_PATH }}
|
||||||
|
key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-pnpm-store-
|
||||||
|
|
||||||
|
- name: Install frontend dependencies
|
||||||
|
run: |
|
||||||
|
pnpm -r install
|
||||||
|
pnpm -r --filter "./easytier-web/*" build
|
||||||
|
|
||||||
- name: Cargo cache
|
- name: Cargo cache
|
||||||
uses: actions/cache@v4
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
|
|||||||
25
Cargo.lock
generated
25
Cargo.lock
generated
@@ -571,6 +571,20 @@ dependencies = [
|
|||||||
"tracing",
|
"tracing",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "axum-embed"
|
||||||
|
version = "0.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "077959a7f8cf438676af90b483304528eb7e16eadadb7f44e9ada4f9dceb9e62"
|
||||||
|
dependencies = [
|
||||||
|
"axum-core",
|
||||||
|
"chrono",
|
||||||
|
"http",
|
||||||
|
"mime_guess",
|
||||||
|
"rust-embed",
|
||||||
|
"tower-service",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "axum-login"
|
name = "axum-login"
|
||||||
version = "0.16.0"
|
version = "0.16.0"
|
||||||
@@ -2028,6 +2042,7 @@ dependencies = [
|
|||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
"axum",
|
"axum",
|
||||||
|
"axum-embed",
|
||||||
"axum-login",
|
"axum-login",
|
||||||
"axum-messages",
|
"axum-messages",
|
||||||
"base64 0.22.1",
|
"base64 0.22.1",
|
||||||
@@ -4116,6 +4131,16 @@ version = "0.3.17"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a"
|
checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "mime_guess"
|
||||||
|
version = "2.0.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f7c44f8e672c00fe5308fa235f821cb4198414e1c77935c1ab6948d3fd78550e"
|
||||||
|
dependencies = [
|
||||||
|
"mime",
|
||||||
|
"unicase",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "minimal-lexical"
|
name = "minimal-lexical"
|
||||||
version = "0.2.1"
|
version = "0.2.1"
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ axum = { version = "0.7", features = ["macros"] }
|
|||||||
axum-login = { version = "0.16" }
|
axum-login = { version = "0.16" }
|
||||||
password-auth = { version = "1.0.0" }
|
password-auth = { version = "1.0.0" }
|
||||||
axum-messages = "0.7.0"
|
axum-messages = "0.7.0"
|
||||||
|
axum-embed = { version = "0.1.0", optional = true }
|
||||||
tower-sessions-sqlx-store = { version = "0.14.1", features = ["sqlite"] }
|
tower-sessions-sqlx-store = { version = "0.14.1", features = ["sqlite"] }
|
||||||
tower-sessions = { version = "0.13.0", default-features = false, features = [
|
tower-sessions = { version = "0.13.0", default-features = false, features = [
|
||||||
"signed",
|
"signed",
|
||||||
@@ -59,3 +60,7 @@ uuid = { version = "1.5.0", features = [
|
|||||||
] }
|
] }
|
||||||
|
|
||||||
chrono = { version = "0.4.37", features = ["serde"] }
|
chrono = { version = "0.4.37", features = ["serde"] }
|
||||||
|
|
||||||
|
[features]
|
||||||
|
default = []
|
||||||
|
embed = ["dep:axum-embed"]
|
||||||
@@ -22,3 +22,9 @@ cli:
|
|||||||
api_server_port:
|
api_server_port:
|
||||||
en: "The port to listen for the restful server, acting as ApiHost and used by the web frontend"
|
en: "The port to listen for the restful server, acting as ApiHost and used by the web frontend"
|
||||||
zh-CN: "restful 服务器的监听端口,作为 ApiHost 并被 web 前端使用"
|
zh-CN: "restful 服务器的监听端口,作为 ApiHost 并被 web 前端使用"
|
||||||
|
web_server_port:
|
||||||
|
en: "The port to listen for the web dashboard server"
|
||||||
|
zh-CN: "web dashboard 服务器的监听端口"
|
||||||
|
no_web:
|
||||||
|
en: "Do not run the web dashboard server"
|
||||||
|
zh-CN: "不运行 web dashboard 服务器"
|
||||||
@@ -5,7 +5,7 @@ extern crate rust_i18n;
|
|||||||
|
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
use clap::{command, Parser};
|
use clap::Parser;
|
||||||
use easytier::{
|
use easytier::{
|
||||||
common::{
|
common::{
|
||||||
config::{ConfigLoader, ConsoleLoggerConfig, FileLoggerConfig, TomlConfigLoader},
|
config::{ConfigLoader, ConsoleLoggerConfig, FileLoggerConfig, TomlConfigLoader},
|
||||||
@@ -21,6 +21,9 @@ mod db;
|
|||||||
mod migrator;
|
mod migrator;
|
||||||
mod restful;
|
mod restful;
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
mod web;
|
||||||
|
|
||||||
rust_i18n::i18n!("locales", fallback = "en");
|
rust_i18n::i18n!("locales", fallback = "en");
|
||||||
|
|
||||||
#[derive(Parser, Debug)]
|
#[derive(Parser, Debug)]
|
||||||
@@ -70,6 +73,23 @@ struct Cli {
|
|||||||
help = t!("cli.api_server_port").to_string(),
|
help = t!("cli.api_server_port").to_string(),
|
||||||
)]
|
)]
|
||||||
api_server_port: u16,
|
api_server_port: u16,
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
#[arg(
|
||||||
|
long,
|
||||||
|
short='l',
|
||||||
|
default_value = "11210",
|
||||||
|
help = t!("cli.web_server_port").to_string(),
|
||||||
|
)]
|
||||||
|
web_server_port: u16,
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
#[arg(
|
||||||
|
long,
|
||||||
|
help = t!("cli.no_web").to_string(),
|
||||||
|
default_value = "false"
|
||||||
|
)]
|
||||||
|
no_web: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_listener_by_url(
|
pub fn get_listener_by_url(
|
||||||
@@ -120,6 +140,20 @@ async fn main() {
|
|||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
restful_server.start().await.unwrap();
|
restful_server.start().await.unwrap();
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
let mut web_server = web::WebServer::new(
|
||||||
|
format!("0.0.0.0:{}", cli.web_server_port).parse().unwrap()
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
if !cli.no_web {
|
||||||
|
web_server.start().await.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
tokio::signal::ctrl_c().await.unwrap();
|
tokio::signal::ctrl_c().await.unwrap();
|
||||||
}
|
}
|
||||||
|
|||||||
39
easytier-web/src/web/mod.rs
Normal file
39
easytier-web/src/web/mod.rs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
use axum::Router;
|
||||||
|
use easytier::common::scoped_task::ScopedTask;
|
||||||
|
use rust_embed::RustEmbed;
|
||||||
|
use std::net::SocketAddr;
|
||||||
|
use axum_embed::ServeEmbed;
|
||||||
|
use tokio::net::TcpListener;
|
||||||
|
|
||||||
|
/// Embed assets for web dashboard, build frontend first
|
||||||
|
#[derive(RustEmbed, Clone)]
|
||||||
|
#[folder = "frontend/dist/"]
|
||||||
|
struct Assets;
|
||||||
|
|
||||||
|
pub struct WebServer {
|
||||||
|
bind_addr: SocketAddr,
|
||||||
|
serve_task: Option<ScopedTask<()>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl WebServer {
|
||||||
|
pub async fn new(bind_addr: SocketAddr) -> anyhow::Result<Self> {
|
||||||
|
Ok(WebServer {
|
||||||
|
bind_addr,
|
||||||
|
serve_task: None,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn start(&mut self) -> Result<(), anyhow::Error> {
|
||||||
|
let listener = TcpListener::bind(self.bind_addr).await?;
|
||||||
|
let service = ServeEmbed::<Assets>::new();
|
||||||
|
let app = Router::new().fallback_service(service);
|
||||||
|
|
||||||
|
let task = tokio::spawn(async move {
|
||||||
|
axum::serve(listener, app).await.unwrap();
|
||||||
|
});
|
||||||
|
|
||||||
|
self.serve_task = Some(task.into());
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user