feat(tvix/eval): implement attrset update (//
) operator
The underlying implementation does a few tricks based on which pair of attrset representations is encountered. Particularly the effect of short-circuiting the empty cases might be relevant in nixpkgs/NixOS, due to the use of lib.optionalAttrs. Change-Id: I22b978b1c69af12926489a71087c6a6219c012f3 Reviewed-on: https://cl.tvl.fyi/c/depot/+/6140 Reviewed-by: sterni <sternenseemann@systemli.org> Tested-by: BuildkiteCI
This commit is contained in:
parent
fa2d250d1a
commit
058e77bab2
13 changed files with 87 additions and 1 deletions
|
@ -136,6 +136,7 @@ impl Compiler {
|
|||
BinOpKind::Mul => OpCode::OpMul,
|
||||
BinOpKind::Div => OpCode::OpDiv,
|
||||
BinOpKind::Equal => OpCode::OpEqual,
|
||||
BinOpKind::Update => OpCode::OpAttrsUpdate,
|
||||
_ => todo!(),
|
||||
};
|
||||
|
||||
|
@ -187,6 +188,10 @@ impl Compiler {
|
|||
// 2. Keys can refer to nested attribute sets.
|
||||
// 3. Attribute sets can (optionally) be recursive.
|
||||
fn compile_attr_set(&mut self, node: rnix::types::AttrSet) -> EvalResult<()> {
|
||||
if node.recursive() {
|
||||
todo!("recursive attribute sets are not yet implemented")
|
||||
}
|
||||
|
||||
let mut count = 0;
|
||||
|
||||
for kv in node.entries() {
|
||||
|
|
|
@ -33,6 +33,7 @@ pub enum OpCode {
|
|||
// Attribute sets
|
||||
OpAttrs(usize),
|
||||
OpAttrPath(usize),
|
||||
OpAttrsUpdate,
|
||||
|
||||
// Lists
|
||||
OpList(usize),
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
{ a = "ok"; }
|
|
@ -0,0 +1 @@
|
|||
{} // { a = "ok"; }
|
|
@ -0,0 +1 @@
|
|||
{ a = "ok"; }
|
|
@ -0,0 +1 @@
|
|||
{ a = "ok"; } // {}
|
|
@ -0,0 +1 @@
|
|||
{ name = "foo"; other = 42; value = "bar"; }
|
|
@ -0,0 +1 @@
|
|||
{ name = "foo"; value = "bar"; } // { other = 42; }
|
|
@ -0,0 +1 @@
|
|||
{ a = 15; b = "works"; }
|
|
@ -0,0 +1 @@
|
|||
{ a = 15; } // { b = "works"; }
|
|
@ -35,7 +35,7 @@ impl Display for NixAttrs {
|
|||
}
|
||||
|
||||
NixAttrs::Map(map) => {
|
||||
for (name, value) in map {
|
||||
for (name, value) in map.iter() {
|
||||
f.write_fmt(format_args!("{} = {}; ", name.ident_str(), value))?;
|
||||
}
|
||||
}
|
||||
|
@ -54,6 +54,59 @@ impl PartialEq for NixAttrs {
|
|||
}
|
||||
|
||||
impl NixAttrs {
|
||||
// Update one attribute set with the values of the other.
|
||||
pub fn update(&self, other: &Self) -> Self {
|
||||
match (self, other) {
|
||||
// Short-circuit on some optimal cases:
|
||||
(NixAttrs::Empty, NixAttrs::Empty) => NixAttrs::Empty,
|
||||
(NixAttrs::Empty, _) => other.clone(),
|
||||
(_, NixAttrs::Empty) => self.clone(),
|
||||
(NixAttrs::KV { .. }, NixAttrs::KV { .. }) => other.clone(),
|
||||
|
||||
// Slightly more advanced, but still optimised updates
|
||||
(NixAttrs::Map(m), NixAttrs::KV { name, value }) => {
|
||||
let mut m = m.clone();
|
||||
m.insert(NixString::NAME, name.clone());
|
||||
m.insert(NixString::VALUE, value.clone());
|
||||
NixAttrs::Map(m)
|
||||
}
|
||||
|
||||
(NixAttrs::KV { name, value }, NixAttrs::Map(m)) => {
|
||||
let mut m = m.clone();
|
||||
|
||||
match m.entry(NixString::NAME) {
|
||||
std::collections::btree_map::Entry::Vacant(e) => {
|
||||
e.insert(name.clone());
|
||||
}
|
||||
|
||||
std::collections::btree_map::Entry::Occupied(_) => {
|
||||
/* name from `m` has precedence */
|
||||
}
|
||||
};
|
||||
|
||||
match m.entry(NixString::VALUE) {
|
||||
std::collections::btree_map::Entry::Vacant(e) => {
|
||||
e.insert(value.clone());
|
||||
}
|
||||
|
||||
std::collections::btree_map::Entry::Occupied(_) => {
|
||||
/* value from `m` has precedence */
|
||||
}
|
||||
};
|
||||
|
||||
NixAttrs::Map(m)
|
||||
}
|
||||
|
||||
// Plain merge of maps.
|
||||
(NixAttrs::Map(m1), NixAttrs::Map(m2)) => {
|
||||
let mut m1 = m1.clone();
|
||||
let mut m2 = m2.clone();
|
||||
m1.append(&mut m2);
|
||||
NixAttrs::Map(m1)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Retrieve reference to a mutable map inside of an attrs,
|
||||
/// optionally changing the representation if required.
|
||||
fn map_mut(&mut self) -> &mut BTreeMap<NixString, Value> {
|
||||
|
|
|
@ -71,6 +71,16 @@ impl Value {
|
|||
}),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn as_attrs(self) -> EvalResult<Rc<NixAttrs>> {
|
||||
match self {
|
||||
Value::Attrs(s) => Ok(s),
|
||||
other => Err(Error::TypeError {
|
||||
expected: "set",
|
||||
actual: other.type_of(),
|
||||
}),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Display for Value {
|
||||
|
|
|
@ -117,8 +117,17 @@ impl VM {
|
|||
OpCode::OpNull => self.push(Value::Null),
|
||||
OpCode::OpTrue => self.push(Value::Bool(true)),
|
||||
OpCode::OpFalse => self.push(Value::Bool(false)),
|
||||
|
||||
OpCode::OpAttrs(count) => self.run_attrset(count)?,
|
||||
OpCode::OpAttrPath(count) => self.run_attr_path(count)?,
|
||||
|
||||
OpCode::OpAttrsUpdate => {
|
||||
let rhs = self.pop().as_attrs()?;
|
||||
let lhs = self.pop().as_attrs()?;
|
||||
|
||||
self.push(Value::Attrs(Rc::new(lhs.update(&rhs))))
|
||||
}
|
||||
|
||||
OpCode::OpList(count) => self.run_list(count)?,
|
||||
OpCode::OpInterpolate(count) => self.run_interpolate(count)?,
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue