14 Commits

Author SHA1 Message Date
e25c27f082 wip 2026-03-23 11:43:22 -05:00
f30432c1db Cleanup some code that should've been done earlier (#7)
Reviewed-on: http://192.168.1.227:3000/sfrembling/pallet/pulls/7
Co-authored-by: godsfryingpan <sfrembling@gmail.com>
Co-committed-by: godsfryingpan <sfrembling@gmail.com>
2026-03-23 10:27:26 -06:00
1b34dee81e Update README.md 2026-03-23 10:13:26 -06:00
46abec1237 Increment version 2026-03-23 11:09:19 -05:00
49eaf6dbfc Enable compilers other than gcc (#6)
issue: #1
Reviewed-on: http://192.168.1.227:3000/sfrembling/pallet/pulls/6
Co-authored-by: godsfryingpan <sfrembling@gmail.com>
Co-committed-by: godsfryingpan <sfrembling@gmail.com>
2026-03-23 10:08:35 -06:00
02009762c1 update PKGBUILD with gcc as dep (#5)
issue: #3
Reviewed-on: http://192.168.1.227:3000/sfrembling/pallet/pulls/5
Co-authored-by: godsfryingpan <sfrembling@gmail.com>
Co-committed-by: godsfryingpan <sfrembling@gmail.com>
2026-03-23 09:57:14 -06:00
e52ad0bc24 add --force-recompile flag (#4)
issue: #2
Reviewed-on: http://192.168.1.227:3000/sfrembling/pallet/pulls/4
Co-authored-by: godsfryingpan <sfrembling@gmail.com>
Co-committed-by: godsfryingpan <sfrembling@gmail.com>
2026-03-23 09:55:03 -06:00
218a6306d4 Fix issue with glob pattern 2026-03-22 22:04:16 -05:00
a0c1a4e006 Update to not recompile if code hasn't changed 2026-03-22 21:56:49 -05:00
1dcd456cb2 Add List function 2026-03-22 21:41:38 -05:00
2f7cc9c150 Fix completions for utils and add description 2026-03-22 21:23:45 -05:00
c13d6e54ee Add ability to generate shell completions 2026-03-22 21:05:34 -05:00
7000ecb3bf create PKGBUILD for Arch 2026-03-22 19:16:22 -05:00
1d3b12bf2f Update README.md 2026-03-22 18:03:12 -06:00
7 changed files with 370 additions and 30 deletions

148
Cargo.lock generated
View File

@@ -52,6 +52,38 @@ dependencies = [
"windows-sys",
]
[[package]]
name = "async-trait"
version = "0.1.89"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9035ad2d096bed7955a320ee7e2230574d28fd3c3a0f186cbea1ff3c7eed5dbb"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "block-buffer"
version = "0.10.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
dependencies = [
"generic-array",
]
[[package]]
name = "bytes"
version = "1.11.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1e748733b7cbc798e1434b6ac524f0c1ff2ab456fe201501e6497c8417a4fc33"
[[package]]
name = "cfg-if"
version = "1.0.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9330f8b2ff13f34540b44e946ef35111825727b38d33286ef986142615121801"
[[package]]
name = "clap"
version = "4.6.0"
@@ -74,6 +106,15 @@ dependencies = [
"strsim",
]
[[package]]
name = "clap_complete"
version = "4.6.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "19c9f1dde76b736e3681f28cec9d5a61299cbaae0fce80a68e43724ad56031eb"
dependencies = [
"clap",
]
[[package]]
name = "clap_derive"
version = "4.6.0"
@@ -107,12 +148,51 @@ dependencies = [
"windows-sys",
]
[[package]]
name = "cpufeatures"
version = "0.2.17"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280"
dependencies = [
"libc",
]
[[package]]
name = "crypto-common"
version = "0.1.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "78c8292055d1c1df0cce5d180393dc8cce0abec0a7102adb6c7b1eef6016d60a"
dependencies = [
"generic-array",
"typenum",
]
[[package]]
name = "digest"
version = "0.10.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
dependencies = [
"block-buffer",
"crypto-common",
]
[[package]]
name = "equivalent"
version = "1.0.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f"
[[package]]
name = "generic-array"
version = "0.14.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a"
dependencies = [
"typenum",
"version_check",
]
[[package]]
name = "glob"
version = "0.3.3"
@@ -131,6 +211,12 @@ version = "0.5.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea"
[[package]]
name = "hex"
version = "0.4.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70"
[[package]]
name = "indexmap"
version = "2.13.0"
@@ -147,6 +233,12 @@ version = "1.70.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a6cb138bb79a146c1bd460005623e142ef0181e3d0219cb493e02f7d08a35695"
[[package]]
name = "libc"
version = "0.2.183"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b5b646652bf6661599e1da8901b3b9522896f01e736bad5f723fe7a3a27f899d"
[[package]]
name = "once_cell_polyfill"
version = "1.70.2"
@@ -155,15 +247,23 @@ checksum = "384b8ab6d37215f3c5301a95a4accb5d64aa607f1fcb26a11b5303878451b4fe"
[[package]]
name = "pallet"
version = "1.0.2"
version = "1.0.6"
dependencies = [
"clap",
"clap_complete",
"colored",
"glob",
"serde",
"sha256",
"toml",
]
[[package]]
name = "pin-project-lite"
version = "0.2.17"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a89322df9ebe1c1578d689c92318e070967d1042b512afbe49518723f4e6d5cd"
[[package]]
name = "proc-macro2"
version = "1.0.106"
@@ -221,6 +321,30 @@ dependencies = [
"serde_core",
]
[[package]]
name = "sha2"
version = "0.10.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a7507d819769d01a365ab707794a4084392c824f54a7a6a7862f8c3d0892b283"
dependencies = [
"cfg-if",
"cpufeatures",
"digest",
]
[[package]]
name = "sha256"
version = "1.6.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f880fc8562bdeb709793f00eb42a2ad0e672c4f883bbe59122b926eca935c8f6"
dependencies = [
"async-trait",
"bytes",
"hex",
"sha2",
"tokio",
]
[[package]]
name = "strsim"
version = "0.11.1"
@@ -238,6 +362,16 @@ dependencies = [
"unicode-ident",
]
[[package]]
name = "tokio"
version = "1.50.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "27ad5e34374e03cfffefc301becb44e9dc3c17584f414349ebe29ed26661822d"
dependencies = [
"bytes",
"pin-project-lite",
]
[[package]]
name = "toml"
version = "1.0.7+spec-1.1.0"
@@ -277,6 +411,12 @@ version = "1.0.7+spec-1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f17aaa1c6e3dc22b1da4b6bba97d066e354c7945cac2f7852d4e4e7ca7a6b56d"
[[package]]
name = "typenum"
version = "1.19.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "562d481066bde0658276a35467c4af00bdc6ee726305698a55b86e61d7ad82bb"
[[package]]
name = "unicode-ident"
version = "1.0.24"
@@ -289,6 +429,12 @@ version = "0.2.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821"
[[package]]
name = "version_check"
version = "0.9.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a"
[[package]]
name = "windows-link"
version = "0.2.1"

View File

@@ -1,11 +1,14 @@
[package]
name = "pallet"
version = "1.0.2"
version = "1.0.6"
edition = "2024"
description = "A project manager and build system for C inspired by Rust's Cargo"
[dependencies]
clap = { version = "4.6.0", features = ["derive"] }
clap_complete = "4.6.0"
colored = "3.1.1"
glob = "0.3.3"
serde = { version = "1.0.228", features = ["derive"] }
sha256 = "1.6.0"
toml = "1.0.7"

19
PKGBUILD Normal file
View File

@@ -0,0 +1,19 @@
# Maintainer: Shea Frembling <sfrembling@gmail.com>
pkgname=pallet
pkgver=1.0.6
pkgrel=1
pkgdesc="A simple C project manager inspired by Cargo"
arch=('x86_64')
url=""
license=('MIT')
depends=('gcc')
makedepends=('rust' 'cargo')
source=()
build() {
cargo build --release
}
package() {
install -Dm755 "$startdir/target/release/pallet" "$pkgdir/usr/bin/pallet"
}

View File

@@ -4,6 +4,32 @@ Pallet is a project manager and build system for C inspired by Rust's Cargo.
This is a toy project not meant to be taken very seriously.
## Requirements for Use
[GCC](https://en.wikipedia.org/wiki/GNU_Compiler_Collection) is recommended as the compiler installed for `Pallet` projects.
You can define a different `gcc` compatible compiler instead per-project by editing `Pallet.toml`:
```toml
name = "my-app"
default_build = "debug"
compiler = "clang"
[[build]]
name = "debug"
args = [
"-g",
"-O0",
]
[[build]]
name = "release"
args = [
"-DNDEBUG",
"-O3",
]
```
## Usage
- `pallet new <project>`: initializes a new project at `project`

View File

@@ -4,14 +4,16 @@ use std::{
process::Command,
};
use clap::CommandFactory;
use colored::Colorize;
use glob::glob;
const MAIN_C: &str = include_str!("templates/main.c");
const TEST_C: &str = include_str!("templates/it_works.c");
const GITIGNORE: &str = "target/";
#[derive(clap::Parser)]
#[clap(version)]
#[clap(version, about)]
pub struct App {
#[clap(subcommand)]
command: Subcommand,
@@ -28,6 +30,9 @@ enum Subcommand {
Init,
/// Run the local project
Run {
/// Force recompilation of the project
#[arg(long, short)]
force_recompile: bool,
/// The build mode to use
mode: Option<String>,
/// Arguments to pass to the project binary
@@ -36,59 +41,156 @@ enum Subcommand {
},
/// Build the local project
Build {
/// Force recompilation of the project
#[arg(long, short)]
force_recompile: bool,
/// The build mode to use
mode: Option<String>,
},
/// Clean all in progress files
Clean,
/// Utility functions
Utils {
#[clap(subcommand)]
command: UtilSubcommand,
},
/// Run tests in your project
Test {
/// The build mode to use
mode: Option<String>,
},
/// List available build modes
List,
}
#[derive(clap::Subcommand)]
enum UtilSubcommand {
/// Generate shell completions
Completions {
/// The shell to generate completions for
#[arg(value_enum, long, short)]
shell: ShellCompletions,
},
}
#[derive(Clone, clap::ValueEnum)]
enum ShellCompletions {
Bash,
Fish,
PowerShell,
Zsh,
}
impl App {
pub fn run(self) {
match self.command {
Subcommand::New { name } => match create_project(&name) {
Err(e) => {
Subcommand::New { name } => {
if let Err(e) = create_project(&name) {
eprintln!("Error creating project '{name}': {e}");
std::process::exit(1);
}
_ => {}
},
Subcommand::Init => match create_project(".") {
Err(e) => {
}
Subcommand::Init => {
if let Err(e) = create_project(".") {
eprintln!("Error initializing project: {e}");
std::process::exit(1);
}
_ => {}
},
Subcommand::Run { mode, args } => {
match build(&mode) {
Err(e) => {
}
Subcommand::Run {
mode,
args,
force_recompile,
} => {
if let Err(e) = build(&mode, force_recompile) {
eprintln!("Error building project: {e}");
std::process::exit(1);
}
_ => {}
};
if let Err(e) = run(&mode, args) {
eprintln!("Error running project: {e}");
std::process::exit(1);
}
}
Subcommand::Build { mode } => match build(&mode) {
Err(e) => {
Subcommand::Build {
mode,
force_recompile,
} => {
if let Err(e) = build(&mode, force_recompile) {
eprintln!("Error building project: {e}");
std::process::exit(1);
}
_ => {}
},
Subcommand::Clean => match clean() {
Err(e) => {
}
Subcommand::Clean => {
if let Err(e) = clean() {
eprintln!("Error cleaning project: {e}");
std::process::exit(1);
}
_ => {}
}
Subcommand::Utils { command } => match command {
UtilSubcommand::Completions { shell } => {
let name = env!("CARGO_PKG_NAME");
let mut command = App::command();
match shell {
ShellCompletions::Bash => clap_complete::generate(
clap_complete::shells::Bash,
&mut command,
name,
&mut std::io::stdout(),
),
ShellCompletions::Fish => clap_complete::generate(
clap_complete::shells::Fish,
&mut command,
name,
&mut std::io::stdout(),
),
ShellCompletions::PowerShell => clap_complete::generate(
clap_complete::shells::PowerShell,
&mut command,
name,
&mut std::io::stdout(),
),
ShellCompletions::Zsh => clap_complete::generate(
clap_complete::shells::Zsh,
&mut command,
name,
&mut std::io::stdout(),
),
}
}
},
Subcommand::List => {
if let Err(e) = list() {
eprintln!("Error listing build profiles: {e}");
std::process::exit(1);
}
}
Subcommand::Test { mode } => {
if let Err(e) = test(mode) {
eprintln!("Error running tests: {e}");
std::process::exit(1);
}
}
}
}
}
fn test(mode: Option<String>) -> std::io::Result<()> {
for entry in glob("src/tests/*.c").expect("a valid glob pattern") {
if let Ok(file) = entry {}
}
Ok(())
}
fn list() -> std::io::Result<()> {
let conf = get_config().ok_or(std::io::Error::new(
std::io::ErrorKind::NotFound,
"no Pallet.toml found in local directory",
))?;
for build in conf.build {
println!(" - {}: {:?}", build.name.green().bold(), build.args);
}
Ok(())
}
fn create_project<P: AsRef<Path>>(directory: P) -> std::io::Result<()> {
@@ -143,7 +245,7 @@ fn get_config() -> Option<crate::config::Config> {
toml::from_str(&raw).ok()
}
fn build(mode: &Option<String>) -> std::io::Result<()> {
fn build(mode: &Option<String>, force_recompile: bool) -> std::io::Result<()> {
let conf = match get_config() {
Some(conf) => conf,
None => {
@@ -168,7 +270,23 @@ fn build(mode: &Option<String>) -> std::io::Result<()> {
std::fs::create_dir_all(format!("target/{}", build_config.name))?;
let mut command = Command::new("gcc");
let hash = hash_src_tree()?;
let old_compute_path = PathBuf::from(format!("target/{}/.build_hash", build_config.name));
if old_compute_path.exists() && !force_recompile {
let text = std::fs::read_to_string(old_compute_path)?;
if hash.trim() == text.trim() {
println!(
" {} (code not changed, recompile not made)",
"Finished".green().bold()
);
return Ok(());
}
}
let mut command = Command::new(conf.compiler.as_deref().unwrap_or("gcc"));
for arg in &build_config.args {
command.arg(arg);
@@ -186,9 +304,16 @@ fn build(mode: &Option<String>) -> std::io::Result<()> {
.arg("-o")
.arg(format!("target/{}/{}", build_config.name, conf.name));
let mut child = command.spawn()?;
let status = command.status()?;
child.wait()?;
if !status.success() {
return Err(std::io::Error::new(
std::io::ErrorKind::Other,
format!("compiler exited with status {status}"),
));
}
std::fs::write(format!("target/{}/.build_hash", build_config.name), hash)?;
let stop = start.elapsed();
@@ -203,6 +328,18 @@ fn build(mode: &Option<String>) -> std::io::Result<()> {
Ok(())
}
fn hash_src_tree() -> std::io::Result<String> {
let mut hashes = String::new();
for entry in glob("src/**/*").expect("a valid glob pattern") {
if let Ok(file) = entry {
let text = std::fs::read_to_string(file)?;
let hash = sha256::digest(text);
hashes.push_str(hash.as_str());
}
}
Ok(sha256::digest(hashes))
}
fn run(mode: &Option<String>, args: Option<Vec<String>>) -> std::io::Result<()> {
let conf = match get_config() {
Some(conf) => conf,
@@ -240,7 +377,7 @@ fn run(mode: &Option<String>, args: Option<Vec<String>>) -> std::io::Result<()>
}
fn clean() -> std::io::Result<()> {
if let None = get_config() {
if get_config().is_none() {
return Err(std::io::Error::new(
std::io::ErrorKind::NotFound,
"no Pallet.toml found in local dir",

View File

@@ -1,5 +1,7 @@
#[derive(serde::Deserialize, serde::Serialize, Default)]
pub struct Config {
/// The C compiler to use (defaults to "gcc")
pub compiler: Option<String>,
/// The name of the output binary
pub name: String,
/// The default build to use

7
src/templates/it_works.c Normal file
View File

@@ -0,0 +1,7 @@
#include <assert.h>
int main() {
assert((1 + 1) == 2);
return 0;
}