use super::*;
use crate::seqstring::global_string;
use crate::stack::{pop, push};
use crate::value::{Value, VariantData};
use std::sync::Arc;
#[test]
fn test_variant_field_count() {
unsafe {
let variant = Value::Variant(Arc::new(VariantData::new(
global_string("TestTag".to_string()),
vec![Value::Int(10), Value::Int(20), Value::Int(30)],
)));
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, variant);
let stack = variant_field_count(stack);
let (_stack, result) = pop(stack);
assert_eq!(result, Value::Int(3));
}
}
#[test]
fn test_variant_tag() {
unsafe {
let variant = Value::Variant(Arc::new(VariantData::new(
global_string("MyTag".to_string()),
vec![Value::Int(10)],
)));
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, variant);
let stack = variant_tag(stack);
let (_stack, result) = pop(stack);
assert_eq!(result, Value::Symbol(global_string("MyTag".to_string())));
}
}
#[test]
fn test_variant_field_at() {
unsafe {
let str1 = global_string("hello".to_string());
let str2 = global_string("world".to_string());
let variant = Value::Variant(Arc::new(VariantData::new(
global_string("TestTag".to_string()),
vec![
Value::String(str1.clone()),
Value::Int(42),
Value::String(str2.clone()),
],
)));
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, variant.clone());
let stack = push(stack, Value::Int(0));
let stack = variant_field_at(stack);
let (_stack, result) = pop(stack);
assert_eq!(result, Value::String(str1.clone()));
let stack = push(stack, variant.clone());
let stack = push(stack, Value::Int(1));
let stack = variant_field_at(stack);
let (_stack, result) = pop(stack);
assert_eq!(result, Value::Int(42));
let stack = push(stack, variant.clone());
let stack = push(stack, Value::Int(2));
let stack = variant_field_at(stack);
let (_stack, result) = pop(stack);
assert_eq!(result, Value::String(str2));
}
}
#[test]
fn test_variant_field_count_empty() {
unsafe {
let variant = Value::Variant(Arc::new(VariantData::new(
global_string("Empty".to_string()),
vec![],
)));
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, variant);
let stack = variant_field_count(stack);
let (_stack, result) = pop(stack);
assert_eq!(result, Value::Int(0));
}
}
#[test]
fn test_make_variant_with_fields() {
unsafe {
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, Value::Int(10)); let stack = push(stack, Value::Int(20)); let stack = push(stack, Value::Int(30)); let stack = push(stack, Value::Symbol(global_string("Tag".to_string())));
let stack = make_variant_3(stack);
let (_stack, result) = pop(stack);
match result {
Value::Variant(v) => {
assert_eq!(v.tag.as_str(), "Tag");
assert_eq!(v.fields.len(), 3);
assert_eq!(v.fields[0], Value::Int(10));
assert_eq!(v.fields[1], Value::Int(20));
assert_eq!(v.fields[2], Value::Int(30));
}
_ => panic!("Expected Variant"),
}
}
}
#[test]
fn test_make_variant_empty() {
unsafe {
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, Value::Symbol(global_string("None".to_string())));
let stack = make_variant_0(stack);
let (_stack, result) = pop(stack);
match result {
Value::Variant(v) => {
assert_eq!(v.tag.as_str(), "None");
assert_eq!(v.fields.len(), 0);
}
_ => panic!("Expected Variant"),
}
}
}
#[test]
fn test_make_variant_with_mixed_types() {
unsafe {
let s = global_string("hello".to_string());
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, Value::Int(42));
let stack = push(stack, Value::String(s.clone()));
let stack = push(stack, Value::Float(3.5));
let stack = push(stack, Value::Symbol(global_string("Mixed".to_string())));
let stack = make_variant_3(stack);
let (_stack, result) = pop(stack);
match result {
Value::Variant(v) => {
assert_eq!(v.tag.as_str(), "Mixed");
assert_eq!(v.fields.len(), 3);
assert_eq!(v.fields[0], Value::Int(42));
assert_eq!(v.fields[1], Value::String(s));
assert_eq!(v.fields[2], Value::Float(3.5));
}
_ => panic!("Expected Variant"),
}
}
}
#[test]
fn test_variant_append() {
unsafe {
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, Value::Symbol(global_string("Array".to_string()))); let stack = make_variant_0(stack);
let stack = push(stack, Value::Int(42));
let stack = variant_append(stack);
let (_stack, result) = pop(stack);
match result {
Value::Variant(v) => {
assert_eq!(v.tag.as_str(), "Array");
assert_eq!(v.fields.len(), 1);
assert_eq!(v.fields[0], Value::Int(42));
}
_ => panic!("Expected Variant"),
}
}
}
#[test]
fn test_variant_append_multiple() {
unsafe {
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, Value::Symbol(global_string("Object".to_string()))); let stack = make_variant_0(stack);
let key = global_string("name".to_string());
let stack = push(stack, Value::String(key.clone()));
let stack = variant_append(stack);
let val = global_string("John".to_string());
let stack = push(stack, Value::String(val.clone()));
let stack = variant_append(stack);
let (_stack, result) = pop(stack);
match result {
Value::Variant(v) => {
assert_eq!(v.tag.as_str(), "Object");
assert_eq!(v.fields.len(), 2);
assert_eq!(v.fields[0], Value::String(key));
assert_eq!(v.fields[1], Value::String(val));
}
_ => panic!("Expected Variant"),
}
}
}
#[test]
fn test_variant_last() {
unsafe {
let variant = Value::Variant(Arc::new(VariantData::new(
global_string("List".to_string()),
vec![Value::Int(10), Value::Int(20), Value::Int(30)],
)));
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, variant);
let stack = variant_last(stack);
let (_stack, result) = pop(stack);
assert_eq!(result, Value::Int(30));
}
}
#[test]
fn test_variant_init() {
unsafe {
let variant = Value::Variant(Arc::new(VariantData::new(
global_string("Custom".to_string()),
vec![Value::Int(10), Value::Int(20), Value::Int(30)],
)));
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, variant);
let stack = variant_init(stack);
let (_stack, result) = pop(stack);
match result {
Value::Variant(v) => {
assert_eq!(v.tag.as_str(), "Custom"); assert_eq!(v.fields.len(), 2);
assert_eq!(v.fields[0], Value::Int(10));
assert_eq!(v.fields[1], Value::Int(20));
}
_ => panic!("Expected Variant"),
}
}
}
#[test]
fn test_variant_stack_operations() {
unsafe {
let stack = crate::stack::alloc_test_stack();
let stack = push(stack, Value::Symbol(global_string("Stack".to_string()))); let stack = make_variant_0(stack);
let stack = push(stack, Value::Int(10));
let stack = variant_append(stack);
let stack = push(stack, Value::Int(20));
let stack = variant_append(stack);
let (stack, variant) = pop(stack);
let stack = push(stack, variant.clone());
let stack = push(stack, variant);
let stack = variant_last(stack);
let (stack, top) = pop(stack);
assert_eq!(top, Value::Int(20));
let stack = variant_init(stack);
let (stack, variant) = pop(stack);
let stack = push(stack, variant.clone());
let stack = push(stack, variant);
let stack = variant_last(stack);
let (stack, top) = pop(stack);
assert_eq!(top, Value::Int(10));
let (_stack, result) = pop(stack);
match result {
Value::Variant(v) => {
assert_eq!(v.fields.len(), 1);
assert_eq!(v.fields[0], Value::Int(10));
}
_ => panic!("Expected Variant"),
}
}
}
#[test]
fn test_variant_clone_is_o1() {
let mut variant = Value::Variant(Arc::new(VariantData::new(
global_string("Level0".to_string()),
vec![],
)));
for i in 0..100 {
variant = Value::Variant(Arc::new(VariantData::new(
global_string(format!("Level{}", i)),
vec![variant.clone()],
)));
}
let start = std::time::Instant::now();
for _ in 0..1000 {
let _copy = variant.clone();
}
let elapsed = start.elapsed();
assert!(
elapsed.as_millis() < 10,
"Clone took {:?} - should be O(1) with Arc",
elapsed
);
}
#[test]
fn test_variant_arc_sharing() {
let inner = Value::Variant(Arc::new(VariantData::new(
global_string("Inner".to_string()),
vec![Value::Int(42)],
)));
let outer = Value::Variant(Arc::new(VariantData::new(
global_string("Outer".to_string()),
vec![inner.clone()],
)));
let outer_clone = outer.clone();
match (&outer, &outer_clone) {
(Value::Variant(a), Value::Variant(b)) => {
assert_eq!(a.tag, b.tag);
assert_eq!(a.fields.len(), b.fields.len());
}
_ => panic!("Expected Variants"),
}
}
#[test]
fn test_variant_thread_safe_sharing() {
use std::sync::Arc as StdArc;
use std::thread;
let variant = Value::Variant(Arc::new(VariantData::new(
global_string("ThreadSafe".to_string()),
vec![Value::Int(1), Value::Int(2), Value::Int(3)],
)));
let shared = StdArc::new(variant);
let handles: Vec<_> = (0..4)
.map(|_| {
let v = StdArc::clone(&shared);
thread::spawn(move || {
match &*v {
Value::Variant(data) => {
assert_eq!(data.tag.as_str(), "ThreadSafe");
assert_eq!(data.fields.len(), 3);
}
_ => panic!("Expected Variant"),
}
})
})
.collect();
for h in handles {
h.join().expect("Thread panicked");
}
}