extendr_api/lib.rs
1//! An ergonomic, opinionated, safe and user-friendly wrapper to the R-API
2//!
3//! This library aims to provide an interface that will be familiar to
4//! first-time users of Rust or indeed any compiled language.
5//!
6//! See [`Robj`] for much of the content of this crate.
7//! [`Robj`] provides a safe wrapper for the R object type.
8//!
9//! ## Examples
10//!
11//! Use attributes and macros to export to R.
12//!
13//! ```rust,ignore
14//! use extendr_api::prelude::*;
15//! // Export a function or impl to R.
16//! #[extendr]
17//! fn fred(a: i32) -> i32 {
18//! a + 1
19//! }
20//!
21//! // define exports using extendr_module
22//! extendr_module! {
23//! mod mymodule;
24//! fn fred;
25//! }
26//! ```
27//!
28//! In R:
29//!
30//! ```r
31//! result <- fred(1)
32//! ```
33//!
34//! [`Robj`] is a wrapper for R objects.
35//! The [`r!`] and `R!` macros let you build R objects
36//! using Rust and R syntax respectively.
37//! ```
38//! use extendr_api::prelude::*;
39//! test! {
40//! // An R object with a single string "hello"
41//! let character = r!("hello");
42//! let character = r!(["hello", "goodbye"]);
43//!
44//! // An R integer object with a single number 1L.
45//! // Note that in Rust, 1 is an integer and 1.0 is a real.
46//! let integer = r!(1);
47//!
48//! // An R real object with a single number 1.
49//! // Note that in R, 1 is a real and 1L is an integer.
50//! let real = r!(1.0);
51//!
52//! // An R real vector.
53//! let real_vector = r!([1.0, 2.0]);
54//! let real_vector = &[1.0, 2.0].iter().collect_robj();
55//! let real_vector = r!(vec![1.0, 2.0]);
56//!
57//! // An R function object.
58//! let function = R!("function(x, y) { x + y }")?;
59//!
60//! // A named list using the list! macro.
61//! let list = list!(a = 1, b = 2);
62//!
63//! // An unnamed list (of R objects) using the List wrapper.
64//! let list = r!(List::from_values(vec![1, 2, 3]));
65//! let list = r!(List::from_values(vec!["a", "b", "c"]));
66//! let list = r!(List::from_values(&[r!("a"), r!(1), r!(2.0)]));
67//!
68//! // A symbol
69//! let sym = sym!(wombat);
70//!
71//! // A R vector using collect_robj()
72//! let vector = (0..3).map(|x| x * 10).collect_robj();
73//! }
74//! ```
75//!
76//! In Rust, we prefer to use iterators rather than loops.
77//!
78//! ```
79//! use extendr_api::prelude::*;
80//! test! {
81//! // 1 ..= 100 is the same as 1:100
82//! let res = r!(1 ..= 100);
83//! assert_eq!(res, R!("1:100")?);
84//!
85//! // Rust arrays are zero-indexed so it is more common to use 0 .. 100.
86//! let res = r!(0 .. 100);
87//! assert_eq!(res.len(), 100);
88//!
89//! // Using map is a super fast way to generate vectors.
90//! let iter = (0..3).map(|i| format!("fred{}", i));
91//! let character = iter.collect_robj();
92//! assert_eq!(character, r!(["fred0", "fred1", "fred2"]));
93//! }
94//! ```
95//!
96//! To index a vector, first convert it to a slice and then
97//! remember to use 0-based indexing. In Rust, going out of bounds
98//! will cause and error (a panic) unlike C++ which may crash.
99//! ```
100//! use extendr_api::prelude::*;
101//! test! {
102//! let vals = r!([1.0, 2.0]);
103//! let slice = vals.as_real_slice().ok_or("expected slice")?;
104//! let one = slice[0];
105//! let two = slice[1];
106//! // let error = slice[2];
107//! assert_eq!(one, 1.0);
108//! assert_eq!(two, 2.0);
109//! }
110//! ```
111//!
112//! Much slower, but more general are these methods:
113//! ```
114//! use extendr_api::prelude::*;
115//! test! {
116//! let vals = r!([1.0, 2.0, 3.0]);
117//!
118//! // one-based indexing [[i]], returns an object.
119//! assert_eq!(vals.index(1)?, r!(1.0));
120//!
121//! // one-based slicing [x], returns an object.
122//! assert_eq!(vals.slice(1..=2)?, r!([1.0, 2.0]));
123//!
124//! // $ operator, returns an object
125//! let list = list!(a = 1.0, b = "xyz");
126//! assert_eq!(list.dollar("a")?, r!(1.0));
127//! }
128//! ```
129//!
130//! The [`R!`] macro lets you embed R code in Rust
131//! and takes Rust expressions in `{{ }}` pairs.
132//!
133//! The [`Rraw!`] macro will not expand the `{{ }}` pairs.
134//! ```
135//! use extendr_api::prelude::*;
136//! test! {
137//! // The text "1 + 1" is parsed as R source code.
138//! // The result is 1.0 + 1.0 in Rust.
139//! assert_eq!(R!("1 + 1")?, r!(2.0));
140//!
141//! let a = 1.0;
142//! assert_eq!(R!("1 + {{a}}")?, r!(2.0));
143//!
144//! assert_eq!(R!(r"
145//! x <- {{ a }}
146//! x + 1
147//! ")?, r!(2.0));
148//!
149//! assert_eq!(R!(r#"
150//! x <- "hello"
151//! x
152//! "#)?, r!("hello"));
153//!
154//! // Use the R meaning of {{ }} and do not expand.
155//! assert_eq!(Rraw!(r"
156//! x <- {{ 1 }}
157//! x + 1
158//! ")?, r!(2.0));
159//! }
160//! ```
161//!
162//! The [`r!`] macro converts a rust object to an R object
163//! and takes parameters.
164//! ```
165//! use extendr_api::prelude::*;
166//! test! {
167//! // The text "1.0+1.0" is parsed as Rust source code.
168//! let one = 1.0;
169//! assert_eq!(r!(one+1.0), r!(2.0));
170//! }
171//! ```
172//!
173//! Rust has a concept of "Owned" and "Borrowed" objects.
174//!
175//! Owned objects, such as [`Vec`] and [`String`] allocate memory
176//! which is released when the object lifetime ends.
177//!
178//! Borrowed objects such as `&[i32]` and `&str` are just pointers
179//! to annother object's memory and can't live longer than the
180//! object they reference.
181//!
182//! Borrowed objects are much faster than owned objects and use less
183//! memory but are used only for temporary access.
184//!
185//! When we take a slice of an R vector, for example, we need the
186//! original R object to be alive or the data will be corrupted.
187//!
188//! ```
189//! use extendr_api::prelude::*;
190//! test! {
191//! // robj is an "Owned" object that controls the memory allocated.
192//! let robj = r!([1, 2, 3]);
193//!
194//! // Here slice is a "borrowed" reference to the bytes in robj.
195//! // and cannot live longer than robj.
196//! let slice = robj.as_integer_slice().ok_or("expected slice")?;
197//! assert_eq!(slice.len(), 3);
198//! }
199//! ```
200//!
201//! ## Writing tests
202//!
203//! To test the functions exposed to R, wrap your code in the [`test!`] macro.
204//! This macro starts up the necessary R machinery for tests to work.
205//!
206//! ```rust,no_run
207//! use extendr_api::prelude::*;
208//!
209//! #[extendr]
210//! fn things() -> Strings {
211//! Strings::from_values(vec!["Test", "this"])
212//! }
213//!
214//! // define exports using extendr_module
215//! extendr_module! {
216//! mod mymodule;
217//! fn things;
218//! }
219//!
220//!
221//! #[cfg(test)]
222//! mod test {
223//! use super::*;
224//! use extendr_api::prelude::*;
225//!
226//! #[test]
227//! fn test_simple_function() {
228//! assert_eq!(things().elt(0), "Test")
229//! }
230//! }
231//! ```
232//!
233//! ## Returning `Result<T, E>` to R
234//!
235//! Two experimental features for returning error-aware R `list`s, `result_list` and `result_condition`,
236//! can be toggled to avoid panics on `Err`. Instead, an `Err` `x` is returned as either
237//! - list: `list(ok=NULL, err=x)` when `result_list` is enabled,
238//! - error condition: `<error: extendr_error>`, with `x` placed in `condition$value`, when `resultd_condition` is enabled.
239//!
240//! It is currently solely up to the user to handle any result on R side.
241//!
242//! There is an added overhead of wrapping Rust results in an R `list` object.
243//!
244//! ```rust,ignore
245//! use extendr_api::prelude::*;
246//! // simple function always returning an Err string
247//! #[extendr]
248//! fn oups(a: i32) -> std::result::Result<i32, String> {
249//! Err("I did it again".to_string())
250//! }
251//!
252//! // define exports using extendr_module
253//! extendr_module! {
254//! mod mymodule;
255//! fn oups;
256//! }
257//!
258//! ```
259//!
260//! In R:
261//!
262//! ```rust,ignore
263//! # default result_panic feature
264//! oups(1)
265//! > ... long panic traceback from rust printed to stderr
266//!
267//! # result_list feature
268//! lst <- oups(1)
269//! print(lst)
270//! > list(ok = NULL, err = "I did it again")
271//!
272//! # result_condition feature
273//! cnd <- oups(1)
274//! print(cnd)
275//! > <error: extendr_error>
276//! print(cnd$value)
277//! > "I did it again"
278//!
279//! # handling example for result_condition
280//! oups_handled <- function(a) {
281//! val_or_err <- oups(1)
282//! if (inherits(val_or_err, "extendr_error")) stop(val_or_err)
283//! val_or_err
284//! }
285//! ```
286//!
287//! ## Feature gates
288//!
289//! extendr-api has some optional features behind these feature gates:
290//!
291//! - `ndarray`: provides the conversion between R's matrices and [`ndarray`](https://docs.rs/ndarray/latest/ndarray/).
292//! - `num-complex`: provides the conversion between R's complex numbers and [`num-complex`](https://docs.rs/num-complex/latest/num_complex/).
293//! - `serde`: provides the [`serde`](https://serde.rs/) support.
294//! - `graphics`: provides the functionality to control or implement graphics devices.
295//! - `either`: provides implementation of type conversion traits for `Either<L, R>` from [`either`](https://docs.rs/either/latest/either/) if `L` and `R` both implement those traits.
296//! - `faer`: provides conversion between R's matrices and [`faer`](https://docs.rs/faer/latest/faer/).
297//!
298//! extendr-api supports three ways of returning a Result<T,E> to R.
299//! Only one behavior feature can be enabled at a time.
300//! - `result_panic`: Default behavior, return `Ok` as is, panic! on any `Err`
301//!
302//! Default behavior can be overridden by specifying `extend_api` features, i.e. `extendr-api = {..., default-features = false, features= ["result_condition"]}`
303//! These features are experimental and are subject to change.
304//! - `result_list`: return `Ok` as `list(ok=?, err=NULL)` or `Err` `list(ok=NULL, err=?)`
305//! - `result_condition`: return `Ok` as is or `Err` as $value in an R error condition.
306#![doc(
307 html_logo_url = "https://raw.githubusercontent.com/extendr/extendr/main/extendr-logo-256.png"
308)]
309
310pub mod conditions;
311pub mod error;
312pub mod functions;
313pub mod io;
314pub mod iter;
315pub mod lang_macros;
316pub mod metadata;
317pub mod na;
318pub mod optional;
319pub mod ownership;
320pub mod prelude;
321pub mod rmacros;
322pub mod robj;
323pub mod scalar;
324pub mod thread_safety;
325pub mod wrapper;
326
327pub use robj::Robj;
328pub use std::convert::{TryFrom, TryInto};
329pub use std::ops::Deref;
330pub use std::ops::DerefMut;
331
332#[cfg(feature = "serde")]
333pub mod serializer;
334
335#[cfg(feature = "serde")]
336pub mod deserializer;
337
338#[cfg(feature = "graphics")]
339pub mod graphics;
340
341pub(crate) mod conversions;
342
343//////////////////////////////////////////////////
344// Note these pub use statements are deprecated
345//
346// `use extendr_api::prelude::*;`
347//
348// instead.
349
350pub use error::*;
351pub use functions::*;
352pub use lang_macros::*;
353pub use na::*;
354pub use robj::*;
355pub use thread_safety::{catch_r_error, single_threaded, throw_r_error};
356pub use wrapper::*;
357
358pub use extendr_macros::*;
359
360use extendr_ffi::SEXPTYPE;
361use scalar::Rbool;
362
363//////////////////////////////////////////////////
364
365/// TRUE value eg. `r!(TRUE)`
366pub const TRUE: Rbool = Rbool::true_value();
367
368/// FALSE value eg. `r!(FALSE)`
369pub const FALSE: Rbool = Rbool::false_value();
370
371/// NULL value eg. `r!(NULL)`
372pub const NULL: () = ();
373
374/// NA value for integers eg. `r!(NA_INTEGER)`
375pub const NA_INTEGER: Option<i32> = None;
376
377/// NA value for real values eg. `r!(NA_REAL)`
378pub const NA_REAL: Option<f64> = None;
379
380/// NA value for strings. `r!(NA_STRING)`
381pub const NA_STRING: Option<&str> = None;
382
383/// NA value for logical. `r!(NA_LOGICAL)`
384pub const NA_LOGICAL: Rbool = Rbool::na_value();
385
386/// This is needed for the generation of wrappers.
387#[doc(hidden)]
388pub use extendr_ffi::DllInfo;
389
390/// This is necessary for `#[extendr]`-impl
391#[doc(hidden)]
392pub use extendr_ffi::R_ExternalPtrAddr;
393
394/// This is used in `#[extendr(use_rng = true)]` on `fn`-items.
395#[doc(hidden)]
396pub use extendr_ffi::GetRNGstate;
397
398/// This is used in `#[extendr(use_rng = true)]` on `fn`-items.
399#[doc(hidden)]
400pub use extendr_ffi::PutRNGstate;
401
402#[doc(hidden)]
403pub use extendr_ffi::SEXP;
404
405#[doc(hidden)]
406pub use extendr_ffi::{R_NilValue, Rf_errorcall, Rf_warningcall};
407
408pub use metadata::Metadata;
409
410#[doc(hidden)]
411pub struct CallMethod {
412 pub call_symbol: std::ffi::CString,
413 pub func_ptr: *const u8,
414 pub num_args: i32,
415}
416
417unsafe fn make_method_def(
418 cstrings: &mut Vec<std::ffi::CString>,
419 rmethods: &mut Vec<extendr_ffi::R_CallMethodDef>,
420 func: &metadata::Func,
421 wrapped_name: &str,
422) {
423 cstrings.push(std::ffi::CString::new(wrapped_name).unwrap());
424 rmethods.push(extendr_ffi::R_CallMethodDef {
425 name: cstrings.last().unwrap().as_ptr(),
426 fun: Some(std::mem::transmute::<
427 *const u8,
428 unsafe extern "C" fn() -> *mut std::ffi::c_void,
429 >(func.func_ptr)),
430 numArgs: func.args.len() as i32,
431 });
432}
433
434// Internal function used to implement the .Call interface.
435// This is called from the code generated by the #[extendr] attribute.
436#[doc(hidden)]
437pub unsafe fn register_call_methods(info: *mut extendr_ffi::DllInfo, metadata: Metadata) {
438 let mut rmethods = Vec::new();
439 let mut cstrings = Vec::new();
440 for func in metadata.functions {
441 make_method_def(&mut cstrings, &mut rmethods, &func, func.c_name);
442 }
443
444 for imp in metadata.impls {
445 for func in imp.methods {
446 make_method_def(&mut cstrings, &mut rmethods, &func, func.c_name);
447 }
448 }
449
450 rmethods.push(extendr_ffi::R_CallMethodDef {
451 name: std::ptr::null(),
452 fun: None,
453 numArgs: 0,
454 });
455
456 extendr_ffi::R_registerRoutines(
457 info,
458 std::ptr::null(),
459 rmethods.as_ptr(),
460 std::ptr::null(),
461 std::ptr::null(),
462 );
463
464 // This seems to allow both symbols and strings,
465 extendr_ffi::R_useDynamicSymbols(info, extendr_ffi::Rboolean::FALSE);
466 extendr_ffi::R_forceSymbols(info, extendr_ffi::Rboolean::FALSE);
467}
468
469/// Type of R objects used by [Robj::rtype].
470#[derive(Debug, PartialEq)]
471pub enum Rtype {
472 Null, // NILSXP
473 Symbol, // SYMSXP
474 Pairlist, // LISTSXP
475 Function, // CLOSXP
476 Environment, // ENVSXP
477 Promise, // PROMSXP
478 Language, // LANGSXP
479 Special, // SPECIALSXP
480 Builtin, // BUILTINSXP
481 Rstr, // CHARSXP
482 Logicals, // LGLSXP
483 Integers, // INTSXP
484 Doubles, // REALSXP
485 Complexes, // CPLXSXP
486 Strings, // STRSXP
487 Dot, // DOTSXP
488 Any, // ANYSXP
489 List, // VECSXP
490 Expressions, // EXPRSXP
491 Bytecode, // BCODESXP
492 ExternalPtr, // EXTPTRSXP
493 WeakRef, // WEAKREFSXP
494 Raw, // RAWSXP
495 S4, // S4SXP
496 Unknown,
497}
498
499/// Enum use to unpack R objects into their specialist wrappers.
500// Todo: convert all Robj types to wrappers.
501// Note: this only works if the wrappers are all just SEXPs.
502#[derive(Debug, PartialEq)]
503pub enum Rany<'a> {
504 Null(&'a Robj), // NILSXP
505 Symbol(&'a Symbol), // SYMSXP
506 Pairlist(&'a Pairlist), // LISTSXP
507 Function(&'a Function), // CLOSXP
508 Environment(&'a Environment), // ENVSXP
509 Promise(&'a Promise), // PROMSXP
510 Language(&'a Language), // LANGSXP
511 Special(&'a Primitive), // SPECIALSXP
512 Builtin(&'a Primitive), // BUILTINSXP
513 Rstr(&'a Rstr), // CHARSXP
514 Logicals(&'a Logicals), // LGLSXP
515 Integers(&'a Integers), // INTSXP
516 Doubles(&'a Doubles), // REALSXP
517 Complexes(&'a Complexes), // CPLXSXP
518 Strings(&'a Strings), // STRSXP
519 Dot(&'a Robj), // DOTSXP
520 Any(&'a Robj), // ANYSXP
521 List(&'a List), // VECSXP
522 Expressions(&'a Expressions), // EXPRSXP
523 Bytecode(&'a Robj), // BCODESXP
524 ExternalPtr(&'a Robj), // EXTPTRSXP
525 WeakRef(&'a Robj), // WEAKREFSXP
526 Raw(&'a Raw), // RAWSXP
527 S4(&'a S4), // S4SXP
528 Unknown(&'a Robj),
529}
530
531/// Convert extendr's Rtype to R's SEXPTYPE.
532/// Panics if the type is Unknown.
533pub fn rtype_to_sxp(rtype: Rtype) -> SEXPTYPE {
534 use extendr_ffi::SEXPTYPE;
535 match rtype {
536 Rtype::Null => SEXPTYPE::NILSXP,
537 Rtype::Symbol => SEXPTYPE::SYMSXP,
538 Rtype::Pairlist => SEXPTYPE::LISTSXP,
539 Rtype::Function => SEXPTYPE::CLOSXP,
540 Rtype::Environment => SEXPTYPE::ENVSXP,
541 Rtype::Promise => SEXPTYPE::PROMSXP,
542 Rtype::Language => SEXPTYPE::LANGSXP,
543 Rtype::Special => SEXPTYPE::SPECIALSXP,
544 Rtype::Builtin => SEXPTYPE::BUILTINSXP,
545 Rtype::Rstr => SEXPTYPE::CHARSXP,
546 Rtype::Logicals => SEXPTYPE::LGLSXP,
547 Rtype::Integers => SEXPTYPE::INTSXP,
548 Rtype::Doubles => SEXPTYPE::REALSXP,
549 Rtype::Complexes => SEXPTYPE::CPLXSXP,
550 Rtype::Strings => SEXPTYPE::STRSXP,
551 Rtype::Dot => SEXPTYPE::DOTSXP,
552 Rtype::Any => SEXPTYPE::ANYSXP,
553 Rtype::List => SEXPTYPE::VECSXP,
554 Rtype::Expressions => SEXPTYPE::EXPRSXP,
555 Rtype::Bytecode => SEXPTYPE::BCODESXP,
556 Rtype::ExternalPtr => SEXPTYPE::EXTPTRSXP,
557 Rtype::WeakRef => SEXPTYPE::WEAKREFSXP,
558 Rtype::Raw => SEXPTYPE::RAWSXP,
559 #[cfg(not(use_objsxp))]
560 Rtype::S4 => SEXPTYPE::S4SXP,
561 #[cfg(use_objsxp)]
562 Rtype::S4 => SEXPTYPE::OBJSXP,
563 Rtype::Unknown => panic!("attempt to use Unknown Rtype"),
564 }
565}
566
567/// Convert R's SEXPTYPE to extendr's Rtype.
568pub fn sxp_to_rtype(sxptype: SEXPTYPE) -> Rtype {
569 match sxptype {
570 SEXPTYPE::NILSXP => Rtype::Null,
571 SEXPTYPE::SYMSXP => Rtype::Symbol,
572 SEXPTYPE::LISTSXP => Rtype::Pairlist,
573 SEXPTYPE::CLOSXP => Rtype::Function,
574 SEXPTYPE::ENVSXP => Rtype::Environment,
575 SEXPTYPE::PROMSXP => Rtype::Promise,
576 SEXPTYPE::LANGSXP => Rtype::Language,
577 SEXPTYPE::SPECIALSXP => Rtype::Special,
578 SEXPTYPE::BUILTINSXP => Rtype::Builtin,
579 SEXPTYPE::CHARSXP => Rtype::Rstr,
580 SEXPTYPE::LGLSXP => Rtype::Logicals,
581 SEXPTYPE::INTSXP => Rtype::Integers,
582 SEXPTYPE::REALSXP => Rtype::Doubles,
583 SEXPTYPE::CPLXSXP => Rtype::Complexes,
584 SEXPTYPE::STRSXP => Rtype::Strings,
585 SEXPTYPE::DOTSXP => Rtype::Dot,
586 SEXPTYPE::ANYSXP => Rtype::Any,
587 SEXPTYPE::VECSXP => Rtype::List,
588 SEXPTYPE::EXPRSXP => Rtype::Expressions,
589 SEXPTYPE::BCODESXP => Rtype::Bytecode,
590 SEXPTYPE::EXTPTRSXP => Rtype::ExternalPtr,
591 SEXPTYPE::WEAKREFSXP => Rtype::WeakRef,
592 SEXPTYPE::RAWSXP => Rtype::Raw,
593 #[cfg(not(use_objsxp))]
594 SEXPTYPE::S4SXP => Rtype::S4,
595 #[cfg(use_objsxp)]
596 SEXPTYPE::OBJSXP => Rtype::S4,
597 _ => Rtype::Unknown,
598 }
599}
600
601const PRINTF_NO_FMT_CSTRING: &[std::os::raw::c_char] = &[37, 115, 0]; // same as "%s\0"
602#[doc(hidden)]
603pub fn print_r_output<T: Into<Vec<u8>>>(s: T) {
604 let cs = std::ffi::CString::new(s).expect("NulError");
605 unsafe {
606 extendr_ffi::Rprintf(PRINTF_NO_FMT_CSTRING.as_ptr(), cs.as_ptr());
607 }
608}
609
610#[doc(hidden)]
611pub fn print_r_error<T: Into<Vec<u8>>>(s: T) {
612 let cs = std::ffi::CString::new(s).expect("NulError");
613 unsafe {
614 extendr_ffi::REprintf(PRINTF_NO_FMT_CSTRING.as_ptr(), cs.as_ptr());
615 }
616}
617
618#[cfg(test)]
619mod tests {
620 use super::prelude::*;
621 use crate as extendr_api;
622
623 use extendr_macros::extendr;
624 use extendr_macros::extendr_module;
625 use extendr_macros::pairlist;
626
627 #[allow(clippy::too_many_arguments)]
628 #[extendr]
629 pub fn inttypes(a: i8, b: u8, c: i16, d: u16, e: i32, f: u32, g: i64, h: u64) {
630 assert_eq!(a, 1);
631 assert_eq!(b, 2);
632 assert_eq!(c, 3);
633 assert_eq!(d, 4);
634 assert_eq!(e, 5);
635 assert_eq!(f, 6);
636 assert_eq!(g, 7);
637 assert_eq!(h, 8);
638 }
639
640 #[extendr]
641 pub fn floattypes(a: f32, b: f64) {
642 assert_eq!(a, 1.);
643 assert_eq!(b, 2.);
644 }
645
646 #[extendr]
647 pub fn strtypes(a: &str, b: String) {
648 assert_eq!(a, "abc");
649 assert_eq!(b, "def");
650 }
651
652 #[extendr]
653 pub fn vectortypes(a: Vec<i32>, b: Vec<f64>) {
654 assert_eq!(a, [1, 2, 3]);
655 assert_eq!(b, [4., 5., 6.]);
656 }
657
658 #[extendr]
659 pub fn robjtype(a: Robj) {
660 assert_eq!(a, Robj::from(1))
661 }
662
663 #[extendr]
664 pub fn return_u8() -> u8 {
665 123
666 }
667
668 #[extendr]
669 pub fn return_u16() -> u16 {
670 123
671 }
672
673 #[extendr]
674 pub fn return_u32() -> u32 {
675 123
676 }
677
678 #[extendr]
679 pub fn return_u64() -> u64 {
680 123
681 }
682
683 #[extendr]
684 pub fn return_i8() -> i8 {
685 123
686 }
687
688 #[extendr]
689 pub fn return_i16() -> i16 {
690 123
691 }
692
693 #[extendr]
694 pub fn return_i32() -> i32 {
695 123
696 }
697
698 #[extendr]
699 pub fn return_i64() -> i64 {
700 123
701 }
702
703 #[extendr]
704 pub fn return_f32() -> f32 {
705 123.
706 }
707
708 #[extendr]
709 pub fn return_f64() -> f64 {
710 123.
711 }
712
713 #[extendr]
714 pub fn f64_slice(x: &[f64]) -> &[f64] {
715 x
716 }
717
718 #[extendr]
719 pub fn i32_slice(x: &[i32]) -> &[i32] {
720 x
721 }
722
723 #[extendr]
724 pub fn bool_slice(x: &[Rbool]) -> &[Rbool] {
725 x
726 }
727
728 #[extendr]
729 pub fn f64_iter(x: Doubles) -> Doubles {
730 x
731 }
732
733 #[extendr]
734 pub fn i32_iter(x: Integers) -> Integers {
735 x
736 }
737
738 // #[extendr]
739 // pub fn bool_iter(x: Logicals) -> Logicals {
740 // x
741 // }
742
743 #[extendr]
744 pub fn symbol(x: Symbol) -> Symbol {
745 x
746 }
747
748 #[extendr]
749 pub fn matrix(x: RMatrix<f64>) -> RMatrix<f64> {
750 x
751 }
752
753 #[extendr]
754 struct Person {
755 pub name: String,
756 }
757
758 #[extendr]
759 /// impl comment.
760 impl Person {
761 fn new() -> Self {
762 Self {
763 name: "".to_string(),
764 }
765 }
766
767 fn set_name(&mut self, name: &str) {
768 self.name = name.to_string();
769 }
770
771 fn name(&self) -> &str {
772 self.name.as_str()
773 }
774 }
775
776 // see metadata_test for the following comments.
777
778 /// comment #1
779 /// comment #2
780 /**
781 comment #3
782 comment #4
783 **/
784 #[extendr]
785 /// aux_func doc comment.
786 fn aux_func(_person: &Person) {}
787
788 // Macro to generate exports
789 extendr_module! {
790 mod my_module;
791 fn aux_func;
792 impl Person;
793 }
794
795 #[test]
796 fn export_test() {
797 test! {
798 use super::*;
799 // Call the exported functions through their generated C wrappers.
800 unsafe {
801 wrap__inttypes(
802 Robj::from(1).get(),
803 Robj::from(2).get(),
804 Robj::from(3).get(),
805 Robj::from(4).get(),
806 Robj::from(5).get(),
807 Robj::from(6).get(),
808 Robj::from(7).get(),
809 Robj::from(8).get(),
810 );
811 wrap__inttypes(
812 Robj::from(1.).get(),
813 Robj::from(2.).get(),
814 Robj::from(3.).get(),
815 Robj::from(4.).get(),
816 Robj::from(5.).get(),
817 Robj::from(6.).get(),
818 Robj::from(7.).get(),
819 Robj::from(8.).get(),
820 );
821 wrap__floattypes(Robj::from(1.).get(), Robj::from(2.).get());
822 wrap__floattypes(Robj::from(1).get(), Robj::from(2).get());
823 wrap__strtypes(Robj::from("abc").get(), Robj::from("def").get());
824 wrap__vectortypes(
825 Robj::from(&[1, 2, 3] as &[i32]).get(),
826 Robj::from(&[4., 5., 6.] as &[f64]).get(),
827 );
828 wrap__robjtype(Robj::from(1).get());
829
830 // General integer types.
831 assert_eq!(Robj::from_sexp(wrap__return_u8()), Robj::from(123_u8));
832 assert_eq!(Robj::from_sexp(wrap__return_u16()), Robj::from(123));
833 assert_eq!(Robj::from_sexp(wrap__return_u32()), Robj::from(123.));
834 assert_eq!(Robj::from_sexp(wrap__return_u64()), Robj::from(123.));
835 assert_eq!(Robj::from_sexp(wrap__return_i8()), Robj::from(123));
836 assert_eq!(Robj::from_sexp(wrap__return_i16()), Robj::from(123));
837 assert_eq!(Robj::from_sexp(wrap__return_i32()), Robj::from(123));
838 assert_eq!(Robj::from_sexp(wrap__return_i64()), Robj::from(123.));
839
840 // Floating point types.
841 assert_eq!(Robj::from_sexp(wrap__return_f32()), Robj::from(123.));
842 assert_eq!(Robj::from_sexp(wrap__return_f64()), Robj::from(123.));
843 }
844 }
845 }
846
847 #[test]
848 fn class_wrapper_test() {
849 test! {
850 let mut person = Person::new();
851 person.set_name("fred");
852 let robj = r!(person);
853 assert_eq!(robj.check_external_ptr_type::<Person>(), true);
854 let person2 = <&Person>::try_from(&robj).unwrap();
855 assert_eq!(person2.name(), "fred");
856 }
857 }
858
859 #[test]
860 fn slice_test() {
861 test! {
862 unsafe {
863 // #[extendr]
864 // pub fn f64_slice(x: &[f64]) -> &[f64] { x }
865
866 let robj = r!([1., 2., 3.]);
867 assert_eq!(Robj::from_sexp(wrap__f64_slice(robj.get())), robj);
868
869 // #[extendr]
870 // pub fn i32_slice(x: &[i32]) -> &[i32] { x }
871
872 let robj = r!([1, 2, 3]);
873 assert_eq!(Robj::from_sexp(wrap__i32_slice(robj.get())), robj);
874
875 // #[extendr]
876 // pub fn bool_slice(x: &[Rbool]) -> &[Rbool] { x }
877
878 let robj = r!([TRUE, FALSE, TRUE]);
879 assert_eq!(Robj::from_sexp(wrap__bool_slice(robj.get())), robj);
880
881 // #[extendr]
882 // pub fn f64_iter(x: Doubles) -> Doubles { x }
883
884 let robj = r!([1., 2., 3.]);
885 assert_eq!(Robj::from_sexp(wrap__f64_iter(robj.get())), robj);
886
887 // #[extendr]
888 // pub fn i32_iter(x: Integers) -> Integers { x }
889
890 let robj = r!([1, 2, 3]);
891 assert_eq!(Robj::from_sexp(wrap__i32_iter(robj.get())), robj);
892
893 // #[extendr]
894 // pub fn bool_iter(x: Logicals) -> Logicals { x }
895
896 // TODO: reinstate this test.
897 // let robj = r!([TRUE, FALSE, TRUE]);
898 // assert_eq!(Robj::from_sexp(wrap__bool_iter(robj.get())), robj);
899
900 // #[extendr]
901 // pub fn symbol(x: Symbol) -> Symbol { x }
902
903 let robj = sym!(fred);
904 assert_eq!(Robj::from_sexp(wrap__symbol(robj.get())), robj);
905
906 // #[extendr]
907 // pub fn matrix(x: Matrix<&[f64]>) -> Matrix<&[f64]> { x }
908
909 let m = RMatrix::new_matrix(1, 2, |r, c| if r == c {1.0} else {0.});
910 let robj = r!(m);
911 assert_eq!(Robj::from_sexp(wrap__matrix(robj.get())), robj);
912 }
913 }
914 }
915
916 #[test]
917 fn r_output_test() {
918 // R equivalent
919 // > txt_con <- textConnection("test_con", open = "w")
920 // > sink(txt_con)
921 // > cat("Hello world")
922 // > sink()
923 // > close(txt_con)
924 // > expect_equal(test_con, "Hello world")
925 //
926
927 test! {
928 let txt_con = R!(r#"textConnection("test_con", open = "w")"#).unwrap();
929 call!("sink", &txt_con).unwrap();
930 rprintln!("Hello world %%!"); //%% checks printf formatting is off, yields one % if on
931 call!("sink").unwrap();
932 call!("close", &txt_con).unwrap();
933 let result = R!("test_con").unwrap();
934 assert_eq!(result, r!("Hello world %%!"));
935 }
936 }
937
938 #[test]
939 fn test_na_str() {
940 assert_ne!(<&str>::na().as_ptr(), "NA".as_ptr());
941 assert_eq!(<&str>::na(), "NA");
942 assert!(!"NA".is_na());
943 assert!(<&str>::na().is_na());
944 }
945
946 #[test]
947 fn metadata_test() {
948 test! {
949 // Rust interface.
950 let metadata = get_my_module_metadata();
951 assert_eq!(metadata.functions[0].doc, " comment #1\n comment #2\n\n comment #3\n comment #4\n *\n aux_func doc comment.");
952 assert_eq!(metadata.functions[0].rust_name, "aux_func");
953 assert_eq!(metadata.functions[0].mod_name, "aux_func");
954 assert_eq!(metadata.functions[0].r_name, "aux_func");
955 assert_eq!(metadata.functions[0].args[0].name, "_person");
956 assert_eq!(metadata.functions[1].rust_name, "get_my_module_metadata");
957 assert_eq!(metadata.impls[0].name, "Person");
958 assert_eq!(metadata.impls[0].methods.len(), 3);
959
960 // R interface
961 let robj = unsafe { Robj::from_sexp(wrap__get_my_module_metadata()) };
962 let functions = robj.dollar("functions").unwrap();
963 let impls = robj.dollar("impls").unwrap();
964 assert_eq!(functions.len(), 3);
965 assert_eq!(impls.len(), 1);
966 }
967 }
968
969 #[test]
970 fn pairlist_macro_works() {
971 test! {
972 assert_eq!(pairlist!(1, 2, 3), Pairlist::from_pairs(&[("", 1), ("", 2), ("", 3)]));
973 assert_eq!(pairlist!(a=1, 2, 3), Pairlist::from_pairs(&[("a", 1), ("", 2), ("", 3)]));
974 assert_eq!(pairlist!(1, b=2, 3), Pairlist::from_pairs(&[("", 1), ("b", 2), ("", 3)]));
975 assert_eq!(pairlist!(a=1, b=2, c=3), Pairlist::from_pairs(&[("a", 1), ("b", 2), ("c", 3)]));
976 assert_eq!(pairlist!(a=NULL), Pairlist::from_pairs(&[("a", ())]));
977 assert_eq!(pairlist!(), Pairlist::from(()));
978 }
979 }
980
981 #[test]
982 fn big_r_macro_works() {
983 test! {
984 assert_eq!(R!("1")?, r!(1.0));
985 assert_eq!(R!(r"1")?, r!(1.0));
986 assert_eq!(R!(r"
987 x <- 1
988 x
989 ")?, r!(1.0));
990 assert_eq!(R!(r"
991 x <- {{ 1.0 }}
992 x
993 ")?, r!(1.0));
994 assert_eq!(R!(r"
995 x <- {{ (0..4).collect_robj() }}
996 x
997 ")?, r!([0, 1, 2, 3]));
998 assert_eq!(R!(r#"
999 x <- "hello"
1000 x
1001 "#)?, r!("hello"));
1002 assert_eq!(Rraw!(r"
1003 x <- {{ 1 }}
1004 x
1005 ")?, r!(1.0));
1006 }
1007 }
1008}