Initial rlbuild commit.
This commit is contained in:
commit
ad31929f7b
3
.gitignore
vendored
Normal file
3
.gitignore
vendored
Normal file
|
@ -0,0 +1,3 @@
|
|||
target/*
|
||||
out/*
|
||||
cache/*
|
33
Cargo.lock
generated
Normal file
33
Cargo.lock
generated
Normal file
|
@ -0,0 +1,33 @@
|
|||
# This file is automatically @generated by Cargo.
|
||||
# It is not intended for manual editing.
|
||||
version = 4
|
||||
|
||||
[[package]]
|
||||
name = "foldhash"
|
||||
version = "0.1.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d9c4f5dac5e15c24eb999c26181a6ca40b39fe946cbe4c263c7209467bc83af2"
|
||||
|
||||
[[package]]
|
||||
name = "hashbrown"
|
||||
version = "0.15.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bf151400ff0baff5465007dd2f3e717f3fe502074ca563069ce3a6629d07b289"
|
||||
dependencies = [
|
||||
"foldhash",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rlbuild"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"rlisp_library",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rlisp_library"
|
||||
version = "0.1.0"
|
||||
source = "git+https://git.ablecorp.us/able/rlisp.git#a1cef24e513a89f8d4de8a1cb3ede546bc72aeca"
|
||||
dependencies = [
|
||||
"hashbrown",
|
||||
]
|
7
Cargo.toml
Normal file
7
Cargo.toml
Normal file
|
@ -0,0 +1,7 @@
|
|||
[package]
|
||||
name = "rlbuild"
|
||||
version = "0.1.0"
|
||||
edition = "2024"
|
||||
|
||||
[dependencies]
|
||||
rlisp_library = { git="https://git.ablecorp.us/able/rlisp.git"}
|
38
assets/pkg.rl
Normal file
38
assets/pkg.rl
Normal file
|
@ -0,0 +1,38 @@
|
|||
;; ! hbpkg.rli ! ;;
|
||||
;; package name & version, non-optional.
|
||||
(package
|
||||
"example"
|
||||
;; Major Minor Patch
|
||||
(version 1 0 1)
|
||||
:authors ("AbleTheAbove")
|
||||
|
||||
:tags ("a" "b" "c")
|
||||
|
||||
(deps
|
||||
;; first item after name is always source.
|
||||
(lily "https://git.ablecorp.us/lily-org/lily.git"
|
||||
;; semantic version. min & max both optional.
|
||||
;; if version is string, exact match will be used.
|
||||
;; if version is "0.x", then the latest of "0.x.y" will be used
|
||||
:version (:min "0.0.2" :max "0.0.5")
|
||||
)
|
||||
(libexample "./src/lib.hb")
|
||||
(third-dep
|
||||
;; Repo lookup refers to the rlrepo project
|
||||
(repo :core "third-dep")
|
||||
|
||||
)
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
(some_clib "https://example.org/clib.git"
|
||||
:commit "ABCDEFGH" ;; optional
|
||||
:branch "trunk" ;; optional
|
||||
;; optional, probably not required if specifying commit.
|
||||
:hash "abcdefghijklmnopqrstuvwxyz123456"
|
||||
)
|
||||
)
|
||||
)
|
0
assets/system.rl
Normal file
0
assets/system.rl
Normal file
9
readme.md
Normal file
9
readme.md
Normal file
|
@ -0,0 +1,9 @@
|
|||
# RLBuild
|
||||
|
||||
|
||||
## Repos
|
||||
rlbuild was built with the ability to point to multiple repos to fetch packages
|
||||
|
||||
repos have a url in the format
|
||||
|
||||
repos.ablecorp.us/<repo_name>/<?sub_repo_name>/<pkg_name>
|
25
src/environ/ext.rs
Normal file
25
src/environ/ext.rs
Normal file
|
@ -0,0 +1,25 @@
|
|||
use rlisp_library::*;
|
||||
|
||||
|
||||
pub fn extend_environ<'a>(mut env: Environ<'a>) -> Environ<'a> {
|
||||
env.data.insert(
|
||||
"system-install".to_string(),
|
||||
Expr::Func(|_args: &[Expr]| -> Result<Expr, RispError> {
|
||||
println!("probably the kernel is installed by now");
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
|
||||
env.data.insert(
|
||||
"use-repo-path".to_string(),
|
||||
Expr::Func(|_args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let repo_name = _args.first().unwrap();
|
||||
|
||||
println!("{:?}", _args);
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env
|
||||
}
|
14
src/environ/mod.rs
Normal file
14
src/environ/mod.rs
Normal file
|
@ -0,0 +1,14 @@
|
|||
use rlisp_library::*;
|
||||
|
||||
mod ext;
|
||||
mod old;
|
||||
|
||||
pub fn extend_environ<'a>(mut env: Environ<'a>) -> Environ<'a> {
|
||||
let mut env = ext::extend_environ(env);
|
||||
let mut env = old::extend_environ(env);
|
||||
|
||||
|
||||
|
||||
|
||||
env
|
||||
}
|
324
src/environ/old.rs
Normal file
324
src/environ/old.rs
Normal file
|
@ -0,0 +1,324 @@
|
|||
use std::{
|
||||
fs::{self, File},
|
||||
io::Write,
|
||||
};
|
||||
|
||||
use rlisp_library::{
|
||||
Environ,
|
||||
Expr::{self, Bool},
|
||||
RispError, default_env, parse_eval,
|
||||
};
|
||||
|
||||
|
||||
|
||||
pub
|
||||
fn extend_environ<'a>(mut env: Environ<'a>) -> Environ<'a> {
|
||||
env.data.insert(
|
||||
"quit".to_string(),
|
||||
Expr::Func(|_args: &[Expr]| -> Result<Expr, RispError> {
|
||||
// TODO: let this function take in arguments
|
||||
println!("Exiting.");
|
||||
std::process::exit(0);
|
||||
#[allow(unreachable_code)]
|
||||
Err(RispError::Reason("Cannot exit process".to_string()))
|
||||
}),
|
||||
);
|
||||
|
||||
env.data.insert(
|
||||
"print".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let mut stri = String::new();
|
||||
for arg in args {
|
||||
let fmted = match arg {
|
||||
Expr::Str(string) => {
|
||||
let string = string.clone();
|
||||
// string.pop();
|
||||
// string.remove(0);
|
||||
|
||||
format!("{}", string)
|
||||
}
|
||||
_ => {
|
||||
format!("{}", arg)
|
||||
}
|
||||
};
|
||||
|
||||
stri.push_str(&fmted);
|
||||
if args.len() > 1 {
|
||||
stri.push_str(" ")
|
||||
}
|
||||
}
|
||||
print!("{}", stri);
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env.data.insert(
|
||||
"println".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let mut stri = String::new();
|
||||
for arg in args {
|
||||
let fmted = match arg {
|
||||
Expr::Str(string) => {
|
||||
let string = string.clone();
|
||||
// string.pop();
|
||||
// string.remove(0);
|
||||
|
||||
format!("{}", string)
|
||||
}
|
||||
_ => {
|
||||
format!("{}", arg)
|
||||
}
|
||||
};
|
||||
|
||||
stri.push_str(&fmted);
|
||||
if args.len() > 1 {
|
||||
stri.push_str(" ")
|
||||
}
|
||||
}
|
||||
println!("{}", stri);
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env.data.insert(
|
||||
"use-repo".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let repo_name = &args[0].clone();
|
||||
let mut repo_name_str = repo_name.to_string();
|
||||
repo_name_str.remove(0);
|
||||
|
||||
let repo_url = &args[1].clone();
|
||||
let use_sources = &args[2].clone();
|
||||
|
||||
let msg = match use_sources {
|
||||
Bool(b) => match b {
|
||||
true => "build_from_src = true",
|
||||
false => "build_from_src = false",
|
||||
},
|
||||
_ => {
|
||||
panic!("AHHH");
|
||||
}
|
||||
};
|
||||
|
||||
let path = format!("out/system/repos");
|
||||
fs::create_dir_all(path).unwrap();
|
||||
|
||||
let path = format!("out/system/repos/{}.repo", repo_name_str);
|
||||
println!("repo name {} repo url {} {}", repo_name, repo_url, msg);
|
||||
|
||||
let mut file = File::create(path).unwrap();
|
||||
let msg = format!("url = {}\n{}", repo_url, msg);
|
||||
let _ = file.write(msg.as_bytes());
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
env.data.insert(
|
||||
"bootloader-install".to_string(),
|
||||
Expr::Func(|_args: &[Expr]| -> Result<Expr, RispError> {
|
||||
// let loader_name = &args[1].clone();
|
||||
// let mut loader_name_str = loader_name.to_string();
|
||||
// loader_name_str.remove(0);
|
||||
|
||||
let path = format!("out/boot/limine");
|
||||
fs::create_dir_all(path).unwrap();
|
||||
|
||||
// let path = format!("out/boot/limine/config.rl");
|
||||
// let mut file = File::create(path).unwrap();
|
||||
// let _ = file.write_all(b"()");
|
||||
|
||||
let path = format!("out/boot/limine/limine.conf");
|
||||
let mut file = File::create(path).unwrap();
|
||||
let _ = file.write_all(b"");
|
||||
|
||||
let path = format!("out/boot/limine/BOOTX64.EFI");
|
||||
let mut x64_file = File::create(&path).unwrap();
|
||||
std::fs::copy("limine/BOOTX64.EFI", &path).unwrap(); // Copy foo.txt to bar.txt
|
||||
|
||||
let path = format!("out/boot/limine/BOOTAA64.EFI");
|
||||
let mut x64_file = File::create(&path).unwrap();
|
||||
std::fs::copy("limine/BOOTAA64.EFI", &path).unwrap(); // Copy foo.txt to bar.txt
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env.data.insert(
|
||||
"pkg-install".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let repo_name = &args[0].clone();
|
||||
let pkg_name = &args[1].clone();
|
||||
let mut pkg_name_str = pkg_name.to_string();
|
||||
pkg_name_str.remove(0);
|
||||
|
||||
println!(
|
||||
"installing package {} from repo {}",
|
||||
pkg_name_str, repo_name
|
||||
);
|
||||
let path = format!("out/programs/{}", pkg_name_str);
|
||||
fs::create_dir_all(path).unwrap();
|
||||
|
||||
let path = format!("out/programs/{}/src", pkg_name_str);
|
||||
fs::create_dir_all(path).unwrap();
|
||||
|
||||
// If package is from :src repo, copy source files
|
||||
if repo_name.to_string() == ":src" {
|
||||
// Try original name first
|
||||
let src_path = format!("sysdata/programs/{}", pkg_name_str);
|
||||
println!("Checking source path: {}", src_path);
|
||||
|
||||
// If not found, try with dashes converted to underscores
|
||||
let underscored_name = pkg_name_str.replace('-', "_");
|
||||
let src_path_alt = format!("sysdata/programs/{}", underscored_name);
|
||||
|
||||
let (found_path, found_name) = if fs::metadata(&src_path).is_ok() {
|
||||
(src_path, pkg_name_str.clone())
|
||||
} else if fs::metadata(&src_path_alt).is_ok() {
|
||||
println!("Found source directory with underscored name at {}", src_path_alt);
|
||||
(src_path_alt, underscored_name)
|
||||
} else {
|
||||
return Err(RispError::Reason(format!(
|
||||
"Source directory not found for package {} at {} or {}",
|
||||
pkg_name_str, src_path, src_path_alt
|
||||
)));
|
||||
};
|
||||
|
||||
println!("Found source directory at {}", found_path);
|
||||
// Copy all files from src directory
|
||||
for entry in fs::read_dir(&found_path).unwrap() {
|
||||
let entry = entry.unwrap();
|
||||
let path = entry.path();
|
||||
if path.is_file() {
|
||||
let file_name = path.file_name().unwrap().to_str().unwrap();
|
||||
let dest_path = format!("out/programs/{}/{}", pkg_name_str, file_name);
|
||||
println!("Copying {} to {}", path.display(), dest_path);
|
||||
fs::copy(&path, &dest_path).unwrap();
|
||||
}
|
||||
}
|
||||
// Copy contents of src directory if it exists
|
||||
let src_src_path = format!("sysdata/programs/{}/src", found_name);
|
||||
println!("Checking src directory at {}", src_src_path);
|
||||
if fs::metadata(&src_src_path).is_ok() {
|
||||
println!("Found src directory at {}", src_src_path);
|
||||
for entry in fs::read_dir(&src_src_path).unwrap() {
|
||||
let entry = entry.unwrap();
|
||||
let path = entry.path();
|
||||
if path.is_file() {
|
||||
let file_name = path.file_name().unwrap().to_str().unwrap();
|
||||
let dest_path = format!("out/programs/{}/src/{}", pkg_name_str, file_name);
|
||||
println!("Copying {} to {}", path.display(), dest_path);
|
||||
fs::copy(&path, &dest_path).unwrap();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Check if meta.rl exists in source, if not generate one
|
||||
let meta_src_path = format!("sysdata/programs/{}/meta.rl", found_name);
|
||||
let meta_dest_path = format!("out/programs/{}/meta.rl", pkg_name_str);
|
||||
|
||||
if !fs::metadata(&meta_src_path).is_ok() {
|
||||
println!("Generating meta.rl for {}", pkg_name_str);
|
||||
let mut file = File::create(&meta_dest_path).unwrap();
|
||||
let meta_content = format!(
|
||||
"(package \"{}\"\n (version 1 0 0)\n :authors (\"\")\n :tags ()\n (deps))\n",
|
||||
pkg_name_str
|
||||
);
|
||||
file.write_all(meta_content.as_bytes()).unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
let path = format!("out/programs/{}/app.axe", pkg_name_str);
|
||||
let mut file = File::create(path).unwrap();
|
||||
let _ = file.write_all(b"");
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env.data.insert(
|
||||
"pkg-configure".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let pkg_name = &args[0].clone();
|
||||
let mut pkg_name_str = pkg_name.to_string();
|
||||
pkg_name_str.remove(0);
|
||||
|
||||
println!("installing package {}.", pkg_name_str);
|
||||
let path = format!("out/programs/{}/config.rl", pkg_name_str);
|
||||
|
||||
println!("configuring package {}", pkg_name_str);
|
||||
// TODO: build the code with the hblang compiler.
|
||||
// TODO: use the meta.rli to map dependencies.
|
||||
let mut file = File::create(path).unwrap();
|
||||
|
||||
|
||||
let _ = file.write_all(b"()");
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env.data.insert(
|
||||
"drivers".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let mut stri = String::new();
|
||||
for arg in args {
|
||||
let fmted = match arg {
|
||||
Expr::Str(string) => {
|
||||
let string = string.clone();
|
||||
format!("{}", string)
|
||||
}
|
||||
_ => {
|
||||
format!("{}", arg)
|
||||
}
|
||||
};
|
||||
|
||||
stri.push_str(&fmted);
|
||||
if args.len() > 1 {
|
||||
stri.push_str(" ")
|
||||
}
|
||||
}
|
||||
println!("Drivers {}", stri);
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
env.data.insert(
|
||||
"services".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let mut stri = String::new();
|
||||
for arg in args {
|
||||
let fmted = match arg {
|
||||
Expr::Str(string) => {
|
||||
let string = string.clone();
|
||||
format!("{}", string)
|
||||
}
|
||||
_ => {
|
||||
format!("{}", arg)
|
||||
}
|
||||
};
|
||||
|
||||
stri.push_str(&fmted);
|
||||
if args.len() > 1 {
|
||||
stri.push_str(" ")
|
||||
}
|
||||
}
|
||||
println!("Services {}", stri);
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env.data.insert(
|
||||
"reincarnation-server".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
let reinc = &args[0].clone();
|
||||
println!("Reincarnation Server {}", reinc);
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env
|
||||
}
|
182
src/environ/pkg_environ.rs
Normal file
182
src/environ/pkg_environ.rs
Normal file
|
@ -0,0 +1,182 @@
|
|||
use std::{fs, File};
|
||||
use rlisp_library::{Environ, Expr, RispError, parse_eval};
|
||||
use crate::packages::{Package, Dependency};
|
||||
|
||||
pub fn extend_environ<'a>(mut env: Environ<'a>) -> Environ<'a> {
|
||||
// Function to create a package
|
||||
env.data.insert(
|
||||
"package".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
if args.len() < 5 {
|
||||
return Err(RispError::Reason("package requires name, authors, version, dependencies, and build command".to_string()));
|
||||
}
|
||||
|
||||
let name = match &args[0] {
|
||||
Expr::Str(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("package name must be a string".to_string())),
|
||||
};
|
||||
|
||||
let authors = match &args[1] {
|
||||
Expr::List(l) => l.iter()
|
||||
.map(|a| match a {
|
||||
Expr::Str(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("authors must be strings".to_string())),
|
||||
})
|
||||
.collect(),
|
||||
_ => return Err(RispError::Reason("authors must be a list".to_string())),
|
||||
};
|
||||
|
||||
let version = match &args[2] {
|
||||
Expr::Str(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("version must be a string".to_string())),
|
||||
};
|
||||
|
||||
let dependencies = match &args[3] {
|
||||
Expr::List(l) => l.iter()
|
||||
.map(|d| match d {
|
||||
Expr::List(dep_list) => {
|
||||
if dep_list.len() != 2 {
|
||||
return Err(RispError::Reason("dependency must have type and source".to_string()));
|
||||
}
|
||||
let dep_type = match &dep_list[0] {
|
||||
Expr::Symbol(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("dependency type must be a symbol".to_string())),
|
||||
};
|
||||
let source = match &dep_list[1] {
|
||||
Expr::Str(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("dependency source must be a string".to_string())),
|
||||
};
|
||||
match dep_type.as_str() {
|
||||
"library" => Ok(Dependency::Library { source }),
|
||||
"binary" => Ok(Dependency::Binary { source }),
|
||||
"source" => Ok(Dependency::Source { source }),
|
||||
_ => Err(RispError::Reason(format!("Unknown dependency type: {}", dep_type))),
|
||||
}
|
||||
}
|
||||
_ => Err(RispError::Reason("dependency must be a list".to_string())),
|
||||
})
|
||||
.collect::<Result<Vec<Dependency>, RispError>>()?,
|
||||
_ => return Err(RispError::Reason("dependencies must be a list".to_string())),
|
||||
};
|
||||
|
||||
let build_command = match &args[4] {
|
||||
Expr::Str(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("build command must be a string".to_string())),
|
||||
};
|
||||
|
||||
let package = Package::new(name, authors, version, dependencies, build_command);
|
||||
|
||||
// Store the package in the environment
|
||||
env.data.insert(
|
||||
"current_package".to_string(),
|
||||
Expr::Custom(Box::new(package))
|
||||
);
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
// Function to get the current package
|
||||
env.data.insert(
|
||||
"get-package".to_string(),
|
||||
Expr::Func(|_args: &[Expr]| -> Result<Expr, RispError> {
|
||||
match env.data.get("current_package") {
|
||||
Some(pkg) => Ok(pkg.clone()),
|
||||
None => Err(RispError::Reason("No package is currently defined".to_string())),
|
||||
}
|
||||
}),
|
||||
);
|
||||
|
||||
// Package installation function
|
||||
env.data.insert(
|
||||
"pkg-install".to_string(),
|
||||
Expr::Func(|args: &[Expr]| -> Result<Expr, RispError> {
|
||||
if args.len() != 2 {
|
||||
return Err(RispError::Reason("pkg-install requires repo name and package name".to_string()));
|
||||
}
|
||||
|
||||
let repo_name = match &args[0] {
|
||||
Expr::Str(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("repo name must be a string".to_string())),
|
||||
};
|
||||
|
||||
let pkg_name = match &args[1] {
|
||||
Expr::Str(s) => s.clone(),
|
||||
_ => return Err(RispError::Reason("package name must be a string".to_string())),
|
||||
};
|
||||
|
||||
println!("Installing package {} from repo {}", pkg_name, repo_name);
|
||||
|
||||
// Create package directory structure
|
||||
let pkg_dir = format!("out/programs/{}", pkg_name);
|
||||
fs::create_dir_all(&pkg_dir).unwrap();
|
||||
fs::create_dir_all(format!("{}/src", pkg_dir)).unwrap();
|
||||
|
||||
// If package is from :src repo, copy source files
|
||||
if repo_name == ":src" {
|
||||
// Try original name first
|
||||
let src_path = format!("sysdata/programs/{}", pkg_name);
|
||||
let underscored_name = pkg_name.replace('-', "_");
|
||||
let src_path_alt = format!("sysdata/programs/{}", underscored_name);
|
||||
|
||||
let (found_path, found_name) = if fs::metadata(&src_path).is_ok() {
|
||||
(src_path, pkg_name.clone())
|
||||
} else if fs::metadata(&src_path_alt).is_ok() {
|
||||
println!("Found source directory with underscored name at {}", src_path_alt);
|
||||
(src_path_alt, underscored_name)
|
||||
} else {
|
||||
return Err(RispError::Reason(format!(
|
||||
"Source directory not found for package {} at {} or {}",
|
||||
pkg_name, src_path, src_path_alt
|
||||
)));
|
||||
};
|
||||
|
||||
// Copy all files from source directory
|
||||
for entry in fs::read_dir(&found_path).unwrap() {
|
||||
let entry = entry.unwrap();
|
||||
let path = entry.path();
|
||||
if path.is_file() {
|
||||
let file_name = path.file_name().unwrap().to_str().unwrap();
|
||||
let dest_path = format!("{}/{}", pkg_dir, file_name);
|
||||
println!("Copying {} to {}", path.display(), dest_path);
|
||||
fs::copy(&path, &dest_path).unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
// Copy contents of src directory if it exists
|
||||
let src_src_path = format!("sysdata/programs/{}/src", found_name);
|
||||
if fs::metadata(&src_src_path).is_ok() {
|
||||
for entry in fs::read_dir(&src_src_path).unwrap() {
|
||||
let entry = entry.unwrap();
|
||||
let path = entry.path();
|
||||
if path.is_file() {
|
||||
let file_name = path.file_name().unwrap().to_str().unwrap();
|
||||
let dest_path = format!("{}/src/{}", pkg_dir, file_name);
|
||||
println!("Copying {} to {}", path.display(), dest_path);
|
||||
fs::copy(&path, &dest_path).unwrap();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Read and evaluate meta.rl directly
|
||||
let meta_path = format!("sysdata/programs/{}/meta.rl", found_name);
|
||||
if let Ok(meta_content) = fs::read_to_string(&meta_path) {
|
||||
println!("Evaluating meta.rl for package {}", pkg_name);
|
||||
if let Err(e) = parse_eval(&meta_content, &mut env.clone()) {
|
||||
println!("Warning: Failed to evaluate meta.rl: {}", e);
|
||||
}
|
||||
} else {
|
||||
println!("Warning: No meta.rl found for package {}", pkg_name);
|
||||
}
|
||||
|
||||
// Create empty app.axe file
|
||||
let app_path = format!("{}/app.axe", pkg_dir);
|
||||
File::create(app_path).unwrap();
|
||||
}
|
||||
|
||||
Ok(Expr::Bool(true))
|
||||
}),
|
||||
);
|
||||
|
||||
env
|
||||
}
|
71
src/main.rs
Normal file
71
src/main.rs
Normal file
|
@ -0,0 +1,71 @@
|
|||
#![feature(slice_take)]
|
||||
#![allow(special_module_name)]
|
||||
|
||||
use std::{
|
||||
fs::{self, File},
|
||||
io::Write,
|
||||
};
|
||||
|
||||
use rlisp_library::{
|
||||
Environ,
|
||||
Expr::{self, Bool},
|
||||
RispError, default_env, parse_eval,
|
||||
};
|
||||
|
||||
mod packages;
|
||||
mod environ;
|
||||
use environ::extend_environ;
|
||||
|
||||
fn main() {
|
||||
let env = &mut default_env();
|
||||
let env = &mut extend_environ(env.clone());
|
||||
// let env = &mut
|
||||
|
||||
let cfg = include_str!("../assets/system.rl");
|
||||
|
||||
let mut complete_exprs: Vec<String> = vec![];
|
||||
let mut left_parens = 0;
|
||||
let mut idx_of_first_left_paran = 0;
|
||||
|
||||
for (i, character) in cfg.chars().enumerate() {
|
||||
if character == '(' {
|
||||
if left_parens == 0 {
|
||||
idx_of_first_left_paran = i;
|
||||
}
|
||||
left_parens += 1
|
||||
}
|
||||
|
||||
if character == ')' {
|
||||
left_parens -= 1;
|
||||
if left_parens == 0 {
|
||||
let idx_of_last_right_paran = i + 1;
|
||||
|
||||
complete_exprs
|
||||
.push(cfg[idx_of_first_left_paran..idx_of_last_right_paran].to_string());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if left_parens != 0 {
|
||||
panic!("unmatched parens. Good luck finding them!");
|
||||
}
|
||||
|
||||
// TODO: Mount the disk image here.
|
||||
|
||||
for expr in complete_exprs {
|
||||
match parse_eval(expr, env) {
|
||||
Ok(_res) => {}
|
||||
Err(e) => {
|
||||
panic!("{:?}", e)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
let path = format!("out/system/config.rl");
|
||||
|
||||
let mut file = File::create(path).unwrap();
|
||||
|
||||
let _ = file.write_all(cfg.as_bytes());
|
||||
|
||||
// TODO: unmount the disk image here.
|
||||
}
|
27
src/packages.rs
Normal file
27
src/packages.rs
Normal file
|
@ -0,0 +1,27 @@
|
|||
#[derive(Debug, Clone)]
|
||||
pub struct Package {
|
||||
pub name: String,
|
||||
pub authors: Vec<String>,
|
||||
pub version: String,
|
||||
pub dependencies: Vec<Dependency>,
|
||||
pub build_command: String,
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone)]
|
||||
pub enum Dependency {
|
||||
Library { source: String },
|
||||
Binary { source: String },
|
||||
Source { source: String },
|
||||
}
|
||||
|
||||
impl Package {
|
||||
pub fn new(name: String, authors: Vec<String>, version: String, dependencies: Vec<Dependency>, build_command: String) -> Self {
|
||||
Self {
|
||||
name,
|
||||
authors,
|
||||
version,
|
||||
dependencies,
|
||||
build_command,
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue