feat(tvix/eval): Implement builtins.concatStringsSep
Change-Id: I6e46bcdbf3b5258b60edb017709fee577eb8ec74 Reviewed-on: https://cl.tvl.fyi/c/depot/+/6907 Reviewed-by: tazjin <tazjin@tvl.su> Tested-by: BuildkiteCI
This commit is contained in:
parent
41ddc37725
commit
66a35de3b6
5 changed files with 46 additions and 0 deletions
|
@ -169,6 +169,22 @@ fn pure_builtins() -> Vec<Builtin> {
|
||||||
Ok(Value::List(res.into()))
|
Ok(Value::List(res.into()))
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
Builtin::new(
|
||||||
|
"concatStringsSep",
|
||||||
|
&[true, true],
|
||||||
|
|args: Vec<Value>, vm: &mut VM| {
|
||||||
|
let separator = args[0].to_str()?;
|
||||||
|
let list = args[1].to_list()?;
|
||||||
|
let mut res = String::new();
|
||||||
|
for (i, val) in list.into_iter().enumerate() {
|
||||||
|
if i != 0 {
|
||||||
|
res.push_str(&separator);
|
||||||
|
}
|
||||||
|
res.push_str(&val.force(vm)?.coerce_to_string(CoercionKind::Weak, vm)?);
|
||||||
|
}
|
||||||
|
Ok(res.into())
|
||||||
|
},
|
||||||
|
),
|
||||||
Builtin::new(
|
Builtin::new(
|
||||||
"div",
|
"div",
|
||||||
&[false, false],
|
&[false, false],
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
[ "" "foobarxyzzy" "foo, bar, xyzzy" "foo" "" ]
|
|
@ -0,0 +1,8 @@
|
||||||
|
with builtins;
|
||||||
|
|
||||||
|
[ (concatStringsSep "" [])
|
||||||
|
(concatStringsSep "" ["foo" "bar" "xyzzy"])
|
||||||
|
(concatStringsSep ", " ["foo" "bar" "xyzzy"])
|
||||||
|
(concatStringsSep ", " ["foo"])
|
||||||
|
(concatStringsSep ", " [])
|
||||||
|
]
|
|
@ -377,6 +377,12 @@ impl From<&str> for Value {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<String> for Value {
|
||||||
|
fn from(val: String) -> Self {
|
||||||
|
Self::String(val.into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn type_error(expected: &'static str, actual: &Value) -> ErrorKind {
|
fn type_error(expected: &'static str, actual: &Value) -> ErrorKind {
|
||||||
ErrorKind::TypeError {
|
ErrorKind::TypeError {
|
||||||
expected,
|
expected,
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
//! backing implementations.
|
//! backing implementations.
|
||||||
use smol_str::SmolStr;
|
use smol_str::SmolStr;
|
||||||
use std::hash::Hash;
|
use std::hash::Hash;
|
||||||
|
use std::ops::Deref;
|
||||||
use std::{borrow::Cow, fmt::Display, str::Chars};
|
use std::{borrow::Cow, fmt::Display, str::Chars};
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
|
@ -178,6 +179,20 @@ impl Display for NixString {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl AsRef<str> for NixString {
|
||||||
|
fn as_ref(&self) -> &str {
|
||||||
|
self.as_str()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for NixString {
|
||||||
|
type Target = str;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
self.as_str()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
Loading…
Reference in a new issue