Compare commits
40 commits
Author | SHA1 | Date | |
---|---|---|---|
db523065e0 | |||
3f87d011e7 | |||
26c46227e6 | |||
05d4233d3f | |||
2a9c2cef76 | |||
ee426b1a43 | |||
97f794a36e | |||
0c1f4ba125 | |||
f1a2ae0311 | |||
e2f2015e06 | |||
4bd50bdadf | |||
28d34eb85d | |||
2c8b4776d9 | |||
6aa876047a | |||
9a12ffc5fc | |||
7ab30ce7c3 | |||
6d50289a35 | |||
3941fe6a1f | |||
b853a6acec | |||
5d372a60de | |||
1a89052244 | |||
1f659c13ab | |||
d6ea7a689e | |||
5a5942c480 | |||
70e801e03c | |||
766156841b | |||
da790827d4 | |||
cb915104cb | |||
856cce5c03 | |||
1a5bd83c1e | |||
0a69149418 | |||
2820a223b4 | |||
362a4cd064 | |||
6e8d6b6e63 | |||
4179c0c716 | |||
17860073ec | |||
8019e1e247 | |||
d9d3dd6bf7 | |||
5b2365e925 | |||
|
961c02bcf9 |
13 changed files with 351 additions and 129 deletions
26
.forgejo/workflows/build-image.yml
Normal file
26
.forgejo/workflows/build-image.yml
Normal file
|
@ -0,0 +1,26 @@
|
|||
on:
|
||||
schedule:
|
||||
- cron: '30 1 * * *'
|
||||
workflow_dispatch:
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@v4
|
||||
|
||||
- name: Clear image from cache
|
||||
run: |
|
||||
docker images git.kemitix.net/${{ env.GITHUB_REPOSITORY }} -q | sort -u | xargs -r docker rmi --force
|
||||
docker system prune --force --all
|
||||
|
||||
- name: Build
|
||||
run: docker build . -t git.kemitix.net/${{ env.GITHUB_REPOSITORY }}:latest
|
||||
|
||||
- name: Login
|
||||
run: docker login git.kemitix.net --username kemitix --password ${{ secrets.FORGEJO_TOKEN_WRITE_PACKAGE }}
|
||||
|
||||
- name: Publish
|
||||
run: docker push git.kemitix.net/${{ env.GITHUB_REPOSITORY }}:latest
|
46
.forgejo/workflows/push-next.yml
Normal file
46
.forgejo/workflows/push-next.yml
Normal file
|
@ -0,0 +1,46 @@
|
|||
on:
|
||||
push:
|
||||
branches:
|
||||
- next
|
||||
jobs:
|
||||
test:
|
||||
runs-on: docker
|
||||
container:
|
||||
image:
|
||||
git.kemitix.net/kemitix/rust:latest
|
||||
strategy:
|
||||
matrix:
|
||||
toolchain:
|
||||
- name: stable
|
||||
- name: nightly
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Ignored Files
|
||||
run: check-for-ignored
|
||||
# run: |
|
||||
# IGNORED=$(git ls-files --cached -i --exclude-standard)
|
||||
# if [ -n "$IGNORED" ]
|
||||
# then
|
||||
# echo "Ignored files present:\n$IGNORED"
|
||||
# exit 1
|
||||
# fi
|
||||
|
||||
- name: Machete
|
||||
run: cargo machete
|
||||
|
||||
- name: Format
|
||||
run: cargo +${{ matrix.toolchain.name }} fmt --check
|
||||
|
||||
- name: Clippy
|
||||
run: cargo +${{ matrix.toolchain.name }} clippy
|
||||
|
||||
- name: Test
|
||||
run: cargo +${{ matrix.toolchain.name }} test
|
||||
|
||||
- name: Build
|
||||
run: cargo +${{ matrix.toolchain.name }} build
|
||||
|
||||
- name: Run
|
||||
run: cargo +${{ matrix.toolchain.name }} run
|
|
@ -1,46 +0,0 @@
|
|||
on:
|
||||
push:
|
||||
branches:
|
||||
- next
|
||||
jobs:
|
||||
test:
|
||||
runs-on: docker
|
||||
strategy:
|
||||
matrix:
|
||||
toolchain:
|
||||
- name: stable
|
||||
- name: nightly
|
||||
- name: v1.81.0
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Machete
|
||||
uses: https://git.kemitix.net/kemitix/rust@v2.5.0
|
||||
with:
|
||||
args: cargo machete
|
||||
|
||||
- name: Format
|
||||
uses: https://git.kemitix.net/kemitix/rust@v2.5.0
|
||||
with:
|
||||
args: ${{ matrix.toolchain.name }} cargo fmt --check
|
||||
|
||||
- name: Clippy
|
||||
uses: https://git.kemitix.net/kemitix/rust@v2.5.0
|
||||
with:
|
||||
args: ${{ matrix.toolchain.name }} cargo clippy
|
||||
|
||||
- name: Test
|
||||
uses: https://git.kemitix.net/kemitix/rust@v2.5.0
|
||||
with:
|
||||
args: ${{ matrix.toolchain.name }} cargo test
|
||||
|
||||
- name: Build
|
||||
uses: https://git.kemitix.net/kemitix/rust@v2.5.0
|
||||
with:
|
||||
args: ${{ matrix.toolchain.name }} cargo build
|
||||
|
||||
- name: Run
|
||||
uses: https://git.kemitix.net/kemitix/rust@v2.5.0
|
||||
with:
|
||||
args: ${{ matrix.toolchain.name }} cargo run
|
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -1 +1,2 @@
|
|||
/target
|
||||
Cargo.lock
|
||||
|
|
|
@ -4,3 +4,5 @@ version = "0.1.0"
|
|||
edition = "2021"
|
||||
|
||||
[dependencies]
|
||||
kxio = "5.1"
|
||||
tokio = { version = "1.43", features = ["full"] }
|
||||
|
|
53
Dockerfile
53
Dockerfile
|
@ -1,18 +1,8 @@
|
|||
FROM docker.io/rust:1.84.0-slim-bookworm
|
||||
FROM docker.io/rust:1.84.0-alpine3.21
|
||||
|
||||
# nodejs - runtime used by forgejo/github actions
|
||||
# curl - to download cargo-binstall
|
||||
# clang-16 & mold - faster linkers for rust
|
||||
# pkg-config - required to compile some rust `-sys` packages
|
||||
# libssl-dev - build dependency for git-next
|
||||
# libdbus-1-dev - linux os interop (e.g. desktop notifications)
|
||||
# git - git
|
||||
RUN apt-get update \
|
||||
&& \
|
||||
apt-get satisfy -y "nodejs (>=18.19.0), curl (>=7.88.1), pkg-config (>=1.8.1), libssl-dev (>=3.0.14), git (>=2.39.2), libdbus-1-dev (>= 1.14.10), libtag1-dev (>= 1.13), libtagc0-dev (>= 1.13), xorg-dev (>=7.7), libxcb-shape0-dev (>=0.15), libxcb-xfixes0-dev (>=0.15)" \
|
||||
&& \
|
||||
rm -r /var/lib/apt/lists/*
|
||||
LABEL org.opencontainers.image.source=https://git.kemitix.net/kemitix/rust
|
||||
|
||||
RUN apk add --no-cache curl=8.11.1-r0
|
||||
RUN curl -L https://github.com/cargo-bins/cargo-binstall/releases/download/v1.10.19/cargo-binstall-x86_64-unknown-linux-musl.tgz -o cargo-binstall.tgz && \
|
||||
tar -xzf cargo-binstall.tgz && \
|
||||
rm cargo-binstall.tgz && \
|
||||
|
@ -25,9 +15,42 @@ RUN cargo binstall -y \
|
|||
cargo-mutants@25.0 \
|
||||
release-plz@0.3
|
||||
|
||||
COPY entrypoint.sh /
|
||||
# should be a no-op if the FROM line is up-to-date
|
||||
RUN rustup update stable
|
||||
|
||||
RUN rustup component add --toolchain stable-x86_64-unknown-linux-musl rustfmt clippy
|
||||
|
||||
# install nightly
|
||||
RUN rustup install nightly && rustup component add --toolchain nightly rustfmt clippy
|
||||
|
||||
# nodejs - runtime used by forgejo/github actions
|
||||
# curl - to download cargo-binstall
|
||||
# clang & mold - faster linkers for rust
|
||||
# pkgconfig - required to compile some rust `-sys` packages
|
||||
# openssl-dev - build dependency for git-next
|
||||
# dbus-dev - linux os interop (e.g. desktop notifications)
|
||||
# perl - native-tls(vendored)
|
||||
# git - git
|
||||
RUN apk add --no-cache \
|
||||
bash \
|
||||
nodejs \
|
||||
build-base \
|
||||
pkgconfig \
|
||||
libssl3 \
|
||||
openssl-dev \
|
||||
perl \
|
||||
dbus-dev \
|
||||
git
|
||||
|
||||
# clang \
|
||||
|
||||
# mold \
|
||||
|
||||
# dbus-dev \
|
||||
|
||||
RUN git config --global user.email "action@git.kemitix.net" && \
|
||||
git config --global user.name "ForgeJo Action. See: https://git.kemitix.net/kemitix/rust"
|
||||
|
||||
ENTRYPOINT [ "/entrypoint.sh" ]
|
||||
COPY scripts/ /usr/local/bin/
|
||||
|
||||
WORKDIR /app
|
||||
|
|
76
README.md
76
README.md
|
@ -11,23 +11,25 @@ on: [push]
|
|||
jobs:
|
||||
test:
|
||||
runs-on: docker
|
||||
container:
|
||||
image:
|
||||
git.kemitix.net/kemitix/rust:v3.0.0
|
||||
steps:
|
||||
- uses: https://git.kemitix.net/kemitix/rust@v2.4.0
|
||||
with:
|
||||
args: nightly cargo test
|
||||
- uses: https://git.kemitix.net/kemitix/rust@v2.4.0
|
||||
with:
|
||||
args: v1.79.0 cargo build
|
||||
- name: test with nightly
|
||||
run: cargo +nightly test
|
||||
- name: build with v1.74.1
|
||||
run: cargo +1.74.1 cargo build
|
||||
- name: test with stable
|
||||
run: cargo test
|
||||
```
|
||||
|
||||
The `args` is one of the following:
|
||||
## Toolchains
|
||||
|
||||
- <COMMAND>
|
||||
- [ nightly | stable | v1.xx.x ] <COMMAND>
|
||||
The available toolchain in the image are:
|
||||
|
||||
`COMMAND` is the command you want to run. The optional prefix is the Rust toolchain, or version. Allowed values are `nightly`, `stable` or a Rust version.
|
||||
|
||||
Where the optional prefix is not given, the `stable` toolchain will be used. The `stable` version is currently `1.82.0`.
|
||||
- `nightly`
|
||||
- stable
|
||||
- 1.74.1
|
||||
|
||||
## Contents
|
||||
|
||||
|
@ -40,13 +42,53 @@ Where the optional prefix is not given, the `stable` toolchain will be used. The
|
|||
- cargo-chef
|
||||
- cargo-hack
|
||||
- release-plz
|
||||
- dbus-dev
|
||||
- perl
|
||||
|
||||
## Binary size
|
||||
### Scripts
|
||||
|
||||
To reduce the size of the debug binary, add the following to the `.cargo/config.toml` file in you project.
|
||||
- `check-for-ignored`
|
||||
|
||||
Checks for files that are being tracked by Git but should be ignored according
|
||||
to the `.gitignore` file.
|
||||
|
||||
#### Usage
|
||||
|
||||
```yaml
|
||||
steps:
|
||||
- name: Check for Ignored Files
|
||||
run: check-for-ignored
|
||||
```
|
||||
|
||||
## Caveats
|
||||
|
||||
### openssl
|
||||
|
||||
The alpine linux install doesn't build with this dependency. You can either compile `native-tls` with the `vendored` feature, or not use `openssl`.
|
||||
|
||||
#### vendoered native-tls
|
||||
|
||||
This crate *must* use the `vendored` feature in order to compile in the Alpine Linux image.
|
||||
|
||||
```toml
|
||||
[profile.dev]
|
||||
debug = 0
|
||||
strip = "debuginfo"
|
||||
native-tls = { version = "0.2", features = ["vendored"] }
|
||||
```
|
||||
|
||||
#### Don't use `openssl`
|
||||
|
||||
Check that none of your dependencies require `openssl`:
|
||||
|
||||
```bash
|
||||
cargo tree --edges normal -i openssl
|
||||
```
|
||||
|
||||
This will list the tree of dependencies that are bringing in `openssl`.
|
||||
|
||||
If you do need ssl/tls, try using `rustls`. e.g.
|
||||
|
||||
```toml
|
||||
reqwest = { version = "0.12", default-features = false, features = [
|
||||
"json",
|
||||
"rustls-tls",
|
||||
] }
|
||||
```
|
||||
|
|
|
@ -1,50 +0,0 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
set -e
|
||||
|
||||
echo "INPUT_ARGS: ${INPUT_ARGS}"
|
||||
|
||||
# split input into an array
|
||||
read -ra ARGS <<<"${INPUT_ARGS}"
|
||||
|
||||
# default toolchain
|
||||
TOOLCHAIN="stable"
|
||||
echo "Default toolchain: ${TOOLCHAIN}"
|
||||
|
||||
# if first parameter is 'nightly'...
|
||||
if test "${ARGS[0]}" == "nightly"; then
|
||||
TOOLCHAIN="nightly"
|
||||
ARGS=("${ARGS[@]:1}")
|
||||
fi
|
||||
if test "${ARGS[0]}" == "stable"; then
|
||||
TOOLCHAIN="stable" # redundant as this is the default
|
||||
ARGS=("${ARGS[@]:1}")
|
||||
fi
|
||||
if [[ "${ARGS[0]}" == v1* ]]; then
|
||||
TOOLCHAIN="${ARGS[0]:1}"
|
||||
ARGS=("${ARGS[@]:1}")
|
||||
fi
|
||||
echo "Selected toolchain: ${TOOLCHAIN}"
|
||||
|
||||
echo ">>> Update toolchain"
|
||||
rustup update "${TOOLCHAIN}"
|
||||
echo ">>> Install rustfmt and clippy"
|
||||
rustup component add --toolchain "${TOOLCHAIN}" rustfmt clippy
|
||||
|
||||
if test "${ARGS[0]}" == "cargo";then
|
||||
PRE_COMMAND="cargo +${TOOLCHAIN} "
|
||||
else
|
||||
PRE_COMMAND="${ARGS[0]}"
|
||||
fi
|
||||
ARGS=("${ARGS[@]:1}")
|
||||
|
||||
# ensure toolchain is up-to-date
|
||||
# recombine remaining arguments
|
||||
COMMAND=$(
|
||||
IFS=" "
|
||||
echo "${ARGS[*]}"
|
||||
)
|
||||
|
||||
# execute command
|
||||
echo ">>> ${PRE_COMMAND} ${COMMAND}"
|
||||
${PRE_COMMAND} ${COMMAND}
|
19
justfile
Normal file
19
justfile
Normal file
|
@ -0,0 +1,19 @@
|
|||
image := "git.kemitix.net/kemitix/rust:test"
|
||||
|
||||
build:
|
||||
docker build . -t {{ image }}
|
||||
|
||||
test: build
|
||||
docker run --rm -v $PWD:/app/ {{ image }} cargo test
|
||||
|
||||
clippy: build
|
||||
docker run --rm -v $PWD:/app/ {{ image }} cargo clippy
|
||||
|
||||
run: build
|
||||
docker run --rm -v $PWD:/app/ {{ image }} cargo run
|
||||
|
||||
fmt: build
|
||||
docker run --rm -v $PWD:/app/ {{ image }} cargo fmt
|
||||
|
||||
shell: build
|
||||
docker run --rm -it -v $PWD:/app/ {{ image }} bash
|
10
mise.toml
Normal file
10
mise.toml
Normal file
|
@ -0,0 +1,10 @@
|
|||
[tasks.test]
|
||||
description = "Test the image"
|
||||
run = [
|
||||
"cargo machete",
|
||||
"cargo fmt --check",
|
||||
"cargo clippy",
|
||||
"cargo test",
|
||||
"cargo build",
|
||||
"cargo run"
|
||||
]
|
30
scripts/check-for-ignored
Executable file
30
scripts/check-for-ignored
Executable file
|
@ -0,0 +1,30 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
# Exit on error
|
||||
set -e
|
||||
|
||||
# Check if git is installed
|
||||
if ! command -v git &> /dev/null; then
|
||||
echo "Error: git is not installed"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Check if we're in a git repository
|
||||
if ! git rev-parse --is-inside-work-tree &> /dev/null; then
|
||||
echo "Error: not in a git repository"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo "Checking for tracked files that should be ignored..."
|
||||
|
||||
# Find files that are both tracked and ignored
|
||||
ignored_files=$(git ls-files --cached -i --exclude-standard)
|
||||
|
||||
if [[ -z "$ignored_files" ]]; then
|
||||
echo "No tracked files are marked as ignored"
|
||||
exit 0
|
||||
else
|
||||
echo "The following tracked files are marked as ignored:"
|
||||
echo "$ignored_files"
|
||||
exit 1
|
||||
fi
|
105
src/kxio.rs
Normal file
105
src/kxio.rs
Normal file
|
@ -0,0 +1,105 @@
|
|||
/// This is an example to show fetching a file from a webiste and saving to a file
|
||||
///
|
||||
/// The example consts of:
|
||||
///
|
||||
/// - The main program, in `main()` - demonstrates how to setup `kxio` for use in prod
|
||||
/// - A test module - demonstrates how to use `kxio` in tests
|
||||
/// - sample functions - showing how to use `kxio` the body of your program, and be testable
|
||||
///
|
||||
/// NOTE: running this program with `cargo run --example get` will create and delete the file
|
||||
/// `example-readme.md` in the current directory.
|
||||
use std::path::Path;
|
||||
|
||||
use kxio::fs::FileHandle;
|
||||
|
||||
// #[tokio::main]
|
||||
pub async fn main() -> kxio::Result<()> {
|
||||
// Create a `Net` object for making real network requests.
|
||||
let net: kxio::net::Net = kxio::net::new();
|
||||
|
||||
// Create a `FileSystem` object for accessing files within the current directory.
|
||||
// The object created will return a `PathTraveral` error result if there is an attempt to\
|
||||
// access a file outside of this directory.
|
||||
let current_dir = std::env::current_dir().map_err(kxio::fs::Error::Io)?;
|
||||
let fs: kxio::fs::FileSystem = kxio::fs::new(current_dir);
|
||||
|
||||
// The URL we will fetch - the readme for this library.
|
||||
let url = "https://git.kemitix.net/kemitix/kxio/raw/branch/main/README.md";
|
||||
|
||||
// Create a PathBuf to a file within the directory that the `fs` object has access to.
|
||||
let file_path = fs.base().join("example-readme.md");
|
||||
|
||||
// Create a generic handle for the file. This doesn't open the file, and always succeeds.
|
||||
let path = fs.path(&file_path);
|
||||
|
||||
// Other options are;
|
||||
// `fs.file(&file_path)` - for a file
|
||||
// `fs.dir(&dir_path)` - for a directory
|
||||
|
||||
// Checks if the path exists (whether a file, directory, etc)
|
||||
if path.exists()? {
|
||||
eprintln!("The file {path} already exists. Aborting!");
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
// Passes a reference to the `fs` and `net` objects for use by your program.
|
||||
// Your programs should not know whether they are handling a mock or the real thing.
|
||||
// Any file or network access should be made using these handlers to be properly testable.
|
||||
let file = download_and_save_to_file(url, &file_path, &fs, &net).await?;
|
||||
read_file(&file)?;
|
||||
delete_file(file)?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// An function that uses a `FileSystem` and a `Net` object to interact with the outside world.
|
||||
async fn download_and_save_to_file(
|
||||
url: &str,
|
||||
file_path: &Path,
|
||||
// The filesystem abstraction
|
||||
fs: &kxio::fs::FileSystem,
|
||||
// The network abstraction
|
||||
net: &kxio::net::Net,
|
||||
) -> kxio::Result<FileHandle> {
|
||||
println!("fetching: {url}");
|
||||
|
||||
// Makes a GET request that can be mocked in a test
|
||||
let response = net.get(url).header("key", "value").send().await?;
|
||||
|
||||
// As you can see, we use [reqwest] under the hood.
|
||||
//
|
||||
// If you need to create a more complex request than the [kxio] fluent API allows, you
|
||||
// can create a request using [reqwest] and pass it to [net.send(request)].
|
||||
|
||||
let body = response.text().await?;
|
||||
println!("fetched {} bytes", body.bytes().len());
|
||||
|
||||
// Uses the file system abstraction to create a handle for a file.
|
||||
let file: kxio::fs::PathReal<kxio::fs::FileMarker> = fs.file(file_path);
|
||||
println!("writing file: {file}");
|
||||
// Writes the body to the file.
|
||||
file.write(body)?;
|
||||
|
||||
Ok(file)
|
||||
}
|
||||
|
||||
/// A function that reads the file contents
|
||||
fn read_file(file: &FileHandle) -> kxio::Result<()> {
|
||||
println!("reading file: {file}");
|
||||
|
||||
// Creates a `Reader` which loaded the file into memory.
|
||||
let reader: kxio::fs::Reader = file.reader()?;
|
||||
let contents: &str = reader.as_str()?;
|
||||
println!("{contents}");
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// A function that deletes the file
|
||||
fn delete_file(file: FileHandle) -> kxio::Result<()> {
|
||||
println!("deleting file: {file}");
|
||||
|
||||
file.remove()?;
|
||||
|
||||
Ok(())
|
||||
}
|
16
src/main.rs
16
src/main.rs
|
@ -1,3 +1,17 @@
|
|||
fn main() {
|
||||
//
|
||||
mod kxio;
|
||||
|
||||
fn main() -> Result<(), Box<dyn std::error::Error>> {
|
||||
println!("Hello, world!");
|
||||
|
||||
let rt = tokio::runtime::Runtime::new()?;
|
||||
Ok(rt.block_on(crate::kxio::main())?)
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
#[test]
|
||||
fn passes() {
|
||||
println!("passes okay");
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue