rustpkg: Finish parsing package scripts and finish boilerplate
This commit is contained in:
parent
226b61ba5f
commit
220144b93c
2 changed files with 285 additions and 29 deletions
|
@ -39,7 +39,9 @@ mod util;
|
||||||
struct PackageScript {
|
struct PackageScript {
|
||||||
id: ~str,
|
id: ~str,
|
||||||
name: ~str,
|
name: ~str,
|
||||||
vers: Version
|
vers: Version,
|
||||||
|
crates: ~[~str],
|
||||||
|
deps: ~[(~str, Option<~str>)]
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PackageScript {
|
impl PackageScript {
|
||||||
|
@ -54,6 +56,8 @@ impl PackageScript {
|
||||||
let crate = parse::parse_crate_from_file(&script, ~[], sess);
|
let crate = parse::parse_crate_from_file(&script, ~[], sess);
|
||||||
let mut id = None;
|
let mut id = None;
|
||||||
let mut vers = None;
|
let mut vers = None;
|
||||||
|
let mut crates = ~[];
|
||||||
|
let mut deps = ~[];
|
||||||
|
|
||||||
fn load_pkg_attr(mis: ~[@ast::meta_item]) -> (Option<~str>,
|
fn load_pkg_attr(mis: ~[@ast::meta_item]) -> (Option<~str>,
|
||||||
Option<~str>) {
|
Option<~str>) {
|
||||||
|
@ -78,6 +82,49 @@ impl PackageScript {
|
||||||
(id, vers)
|
(id, vers)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn load_pkg_dep_attr(mis: ~[@ast::meta_item]) -> (Option<~str>,
|
||||||
|
Option<~str>) {
|
||||||
|
let mut url = None;
|
||||||
|
let mut target = None;
|
||||||
|
|
||||||
|
for mis.each |a| {
|
||||||
|
match a.node {
|
||||||
|
ast::meta_name_value(v, ast::spanned {
|
||||||
|
node: ast::lit_str(s),
|
||||||
|
span: _}) => {
|
||||||
|
match v {
|
||||||
|
~"url" => url = Some(*s),
|
||||||
|
~"target" => target = Some(*s),
|
||||||
|
_ => ()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
(url, target)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn load_pkg_crate_attr(mis: ~[@ast::meta_item]) -> Option<~str> {
|
||||||
|
let mut file = None;
|
||||||
|
|
||||||
|
for mis.each |a| {
|
||||||
|
match a.node {
|
||||||
|
ast::meta_name_value(v, ast::spanned {
|
||||||
|
node: ast::lit_str(s),
|
||||||
|
span: _}) => {
|
||||||
|
match v {
|
||||||
|
~"file" => file = Some(*s),
|
||||||
|
_ => ()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
file
|
||||||
|
}
|
||||||
|
|
||||||
for crate.node.attrs.each |a| {
|
for crate.node.attrs.each |a| {
|
||||||
match a.node.value.node {
|
match a.node.value.node {
|
||||||
ast::meta_list(v, mis) => {
|
ast::meta_list(v, mis) => {
|
||||||
|
@ -88,6 +135,24 @@ impl PackageScript {
|
||||||
id = i;
|
id = i;
|
||||||
vers = v;
|
vers = v;
|
||||||
}
|
}
|
||||||
|
~"pkg_dep" => {
|
||||||
|
let (u, t) = load_pkg_dep_attr(mis);
|
||||||
|
|
||||||
|
if u.is_none() {
|
||||||
|
fail ~"pkg_dep attr without a url value";
|
||||||
|
}
|
||||||
|
|
||||||
|
deps.push((u.get(), t));
|
||||||
|
}
|
||||||
|
~"pkg_crate" => {
|
||||||
|
let f = load_pkg_crate_attr(mis);
|
||||||
|
|
||||||
|
if f.is_none() {
|
||||||
|
fail ~"pkg_file attr without a file value";
|
||||||
|
}
|
||||||
|
|
||||||
|
crates.push(f.get());
|
||||||
|
}
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -105,7 +170,9 @@ impl PackageScript {
|
||||||
PackageScript {
|
PackageScript {
|
||||||
id: id,
|
id: id,
|
||||||
name: util::parse_id(id),
|
name: util::parse_id(id),
|
||||||
vers: util::parse_vers(vers)
|
vers: util::parse_vers(vers),
|
||||||
|
crates: crates,
|
||||||
|
deps: deps
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -114,63 +181,214 @@ impl PackageScript {
|
||||||
|
|
||||||
hasher.write_str(self.id + self.vers.to_str());
|
hasher.write_str(self.id + self.vers.to_str());
|
||||||
|
|
||||||
self.name + hasher.result_str() + self.vers.to_str()
|
fmt!("%s-%s-%s", self.name, hasher.result_str(), self.vers.to_str())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn work_dir() -> Path {
|
fn work_dir() -> Path {
|
||||||
util::root().push(self.hash())
|
util::root().push(~"work").push(self.hash())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Ctx {
|
struct Ctx {
|
||||||
cmd: ~str,
|
|
||||||
args: ~[~str],
|
|
||||||
cfgs: ~[~str],
|
cfgs: ~[~str],
|
||||||
prefer: bool
|
prefer: bool
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Ctx {
|
impl Ctx {
|
||||||
fn run() {
|
fn run(cmd: ~str, args: ~[~str]) {
|
||||||
match self.cmd {
|
let root = util::root();
|
||||||
~"build" => self.build(),
|
|
||||||
~"clean" => self.clean(),
|
util::need_dir(&root);
|
||||||
~"install" => self.install(),
|
util::need_dir(&root.push(~"work"));
|
||||||
~"prefer" => self.prefer(),
|
util::need_dir(&root.push(~"lib"));
|
||||||
~"test" => self.test(),
|
util::need_dir(&root.push(~"bin"));
|
||||||
~"uninstall" => self.uninstall(),
|
util::need_dir(&root.push(~"tmp"));
|
||||||
~"unprefer" => self.unprefer(),
|
|
||||||
|
match cmd {
|
||||||
|
~"build" => self.build(args),
|
||||||
|
~"clean" => self.clean(args),
|
||||||
|
~"install" => self.install(args),
|
||||||
|
~"prefer" => self.prefer(args),
|
||||||
|
~"test" => self.test(args),
|
||||||
|
~"uninstall" => self.uninstall(args),
|
||||||
|
~"unprefer" => self.unprefer(args),
|
||||||
_ => fail ~"reached an unhandled command"
|
_ => fail ~"reached an unhandled command"
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build() {
|
fn build(_args: ~[~str]) -> bool {
|
||||||
let script = PackageScript::parse(os::getcwd());
|
let script = PackageScript::parse(os::getcwd());
|
||||||
|
let dir = script.work_dir();
|
||||||
|
let mut success = true;
|
||||||
|
|
||||||
io::println(fmt!("build: %s (v%s)", script.id, script.vers.to_str()));
|
util::need_dir(&dir);
|
||||||
|
util::info(fmt!("building %s v%s (%s)", script.name, script.vers.to_str(),
|
||||||
|
script.id));
|
||||||
|
|
||||||
|
if script.deps.len() >= 1 {
|
||||||
|
util::info(~"installing dependencies..");
|
||||||
|
|
||||||
|
for script.deps.each |&dep| {
|
||||||
|
let (url, target) = dep;
|
||||||
|
|
||||||
|
success = self.install(if target.is_none() { ~[url] }
|
||||||
|
else { ~[url, target.get()] });
|
||||||
|
|
||||||
|
if !success { break; }
|
||||||
|
}
|
||||||
|
|
||||||
|
if !success {
|
||||||
|
util::error(fmt!("building %s v%s failed: a dep wasn't installed",
|
||||||
|
script.name, script.vers.to_str()));
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
util::info(~"installed dependencies");
|
||||||
|
}
|
||||||
|
|
||||||
|
for script.crates.each |&crate| {
|
||||||
|
success = self.compile(&dir, crate, ~[]);
|
||||||
|
|
||||||
|
if !success { break; }
|
||||||
|
}
|
||||||
|
|
||||||
|
if !success {
|
||||||
|
util::error(fmt!("building %s v%s failed: a crate failed to compile",
|
||||||
|
script.name, script.vers.to_str()));
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
util::info(fmt!("built %s v%s", script.name, script.vers.to_str()));
|
||||||
|
|
||||||
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn clean() {
|
fn compile(dir: &Path, crate: ~str, flags: ~[~str]) -> bool {
|
||||||
|
util::info(~"compiling " + crate);
|
||||||
|
|
||||||
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn install() {
|
fn clean(_args: ~[~str]) -> bool {
|
||||||
|
let script = PackageScript::parse(os::getcwd());
|
||||||
|
let dir = script.work_dir();
|
||||||
|
|
||||||
|
util::info(fmt!("cleaning %s v%s (%s)", script.name, script.vers.to_str(),
|
||||||
|
script.id));
|
||||||
|
|
||||||
|
if os::path_is_dir(&dir) {
|
||||||
|
if os::remove_dir(&dir) {
|
||||||
|
util::info(fmt!("cleaned %s v%s", script.name,
|
||||||
|
script.vers.to_str()));
|
||||||
|
} else {
|
||||||
|
util::error(fmt!("cleaning %s v%s failed",
|
||||||
|
script.name, script.vers.to_str()));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
util::info(fmt!("cleaned %s v%s", script.name,
|
||||||
|
script.vers.to_str()));
|
||||||
|
}
|
||||||
|
|
||||||
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn prefer() {
|
fn install(args: ~[~str]) -> bool {
|
||||||
|
let mut success;
|
||||||
|
let mut dir;
|
||||||
|
|
||||||
|
if args.len() < 1 {
|
||||||
|
util::info(~"installing from the cwd");
|
||||||
|
|
||||||
|
dir = os::getcwd();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
let url = args[0];
|
||||||
|
let target = if args.len() >= 2 { Some(args[1]) }
|
||||||
|
else { None };
|
||||||
|
let hasher = hash::default_state();
|
||||||
|
|
||||||
|
hasher.write_str(url);
|
||||||
|
|
||||||
|
if !target.is_none() {
|
||||||
|
hasher.write_str(target.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
dir = util::root().push(~"tmp").push(hasher.result_str());
|
||||||
|
success = self.fetch(&dir, url, target);
|
||||||
|
|
||||||
|
if !success {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let script = PackageScript::parse(dir);
|
||||||
|
dir = script.work_dir();
|
||||||
|
|
||||||
|
util::info(fmt!("installing %s v%s (%s)", script.name, script.vers.to_str(),
|
||||||
|
script.id));
|
||||||
|
|
||||||
|
if script.deps.len() >= 1 {
|
||||||
|
util::info(~"installing dependencies..");
|
||||||
|
|
||||||
|
for script.deps.each |&dep| {
|
||||||
|
let (url, target) = dep;
|
||||||
|
|
||||||
|
success = self.install(if target.is_none() { ~[url] }
|
||||||
|
else { ~[url, target.get()] });
|
||||||
|
|
||||||
|
if !success { break; }
|
||||||
|
}
|
||||||
|
|
||||||
|
if !success {
|
||||||
|
util::error(fmt!("installing %s v%s failed: a dep wasn't installed",
|
||||||
|
script.name, script.vers.to_str()));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
util::info(~"installed dependencies");
|
||||||
|
}
|
||||||
|
|
||||||
|
for script.crates.each |&crate| {
|
||||||
|
success = self.compile(&dir, crate, ~[]);
|
||||||
|
|
||||||
|
if !success { break; }
|
||||||
|
}
|
||||||
|
|
||||||
|
if !success {
|
||||||
|
util::error(fmt!("installing %s v%s failed: a crate failed to compile",
|
||||||
|
script.name, script.vers.to_str()));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
util::info(fmt!("installed %s v%s", script.name,
|
||||||
|
script.vers.to_str()));
|
||||||
|
|
||||||
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn test() {
|
fn fetch(dir: &Path, url: ~str, target: Option<~str>) -> bool {
|
||||||
|
util::info(fmt!("installing from %s", url));
|
||||||
|
|
||||||
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn uninstall() {
|
fn prefer(_args: ~[~str]) -> bool {
|
||||||
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
fn unprefer() {
|
fn test(_args: ~[~str]) -> bool {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
fn uninstall(_args: ~[~str]) -> bool {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
fn unprefer(_args: ~[~str]) -> bool {
|
||||||
|
true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -217,11 +435,9 @@ pub fn main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
Ctx {
|
Ctx {
|
||||||
cmd: cmd,
|
|
||||||
args: args,
|
|
||||||
cfgs: cfgs,
|
cfgs: cfgs,
|
||||||
prefer: prefer
|
prefer: prefer
|
||||||
}.run();
|
}.run(cmd, args);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub use Crate = api::Crate;
|
pub use Crate = api::Crate;
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use core::*;
|
use core::*;
|
||||||
use rustc::metadata::filesearch;
|
use rustc::metadata::filesearch;
|
||||||
use semver::Version;
|
use semver::Version;
|
||||||
use std::net::url;
|
use std::term;
|
||||||
|
|
||||||
pub fn root() -> Path {
|
pub fn root() -> Path {
|
||||||
match filesearch::get_rustpkg_root() {
|
match filesearch::get_rustpkg_root() {
|
||||||
|
@ -40,6 +40,46 @@ pub fn parse_vers(vers: ~str) -> Version {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn need_dir(s: &Path) {
|
||||||
|
if !os::path_is_dir(s) && !os::make_dir(s, 493_i32) {
|
||||||
|
fail fmt!("can't create dir: %s", s.to_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn info(msg: ~str) {
|
||||||
|
let out = io::stdout();
|
||||||
|
|
||||||
|
if term::color_supported() {
|
||||||
|
term::fg(out, term::color_green);
|
||||||
|
out.write_str(~"info: ");
|
||||||
|
term::reset(out);
|
||||||
|
out.write_line(msg);
|
||||||
|
} else { out.write_line(~"info: " + msg); }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn warn(msg: ~str) {
|
||||||
|
let out = io::stdout();
|
||||||
|
|
||||||
|
if term::color_supported() {
|
||||||
|
term::fg(out, term::color_yellow);
|
||||||
|
out.write_str(~"warning: ");
|
||||||
|
term::reset(out);
|
||||||
|
out.write_line(msg);
|
||||||
|
}else { out.write_line(~"warning: " + msg); }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn error(msg: ~str) {
|
||||||
|
let out = io::stdout();
|
||||||
|
|
||||||
|
if term::color_supported() {
|
||||||
|
term::fg(out, term::color_red);
|
||||||
|
out.write_str(~"error: ");
|
||||||
|
term::reset(out);
|
||||||
|
out.write_line(msg);
|
||||||
|
}
|
||||||
|
else { out.write_line(~"error: " + msg); }
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_is_cmd() {
|
fn test_is_cmd() {
|
||||||
assert is_cmd(~"build");
|
assert is_cmd(~"build");
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue