mezzaluna_load_path/
ruby_core.rs

1use std::path::Path;
2
3/// A Ruby load path builder that prepares paths for in-memory Ruby Core and
4/// Ruby Standard Library sources.
5///
6/// Artichoke embeds Ruby sources and native extensions into an in-memory file
7/// system. This in-memory file system is addressed at a virtual mount point,
8/// which must be located within the VM `$LOAD_PATH` so they may be loaded by
9/// the [require] subsystem.
10///
11/// Like the site directories in MRI, these paths are the lowest priority load
12/// paths and should appear at the end of the `$LOAD_PATH`.
13///
14/// Paths earlier in the sequence returned from [`load_path`] have higher
15/// priority.
16///
17/// ```no_run
18/// use mezzaluna_load_path::RubyCore;
19///
20/// let core_loader = RubyCore::new();
21/// // Load path contains 2 entries: one for Ruby Core sources and one for
22/// // Ruby Standard Library sources.
23/// assert_eq!(core_loader.load_path().len(), 2);
24/// ```
25///
26/// [require]: https://ruby-doc.org/core-3.1.2/Kernel.html#method-i-require
27/// [`load_path`]: Self::load_path
28#[derive(Default, Debug, Clone, Copy, PartialEq, Eq)]
29pub struct RubyCore {
30    _private: (),
31}
32
33impl RubyCore {
34    /// Create a new load path builder that reads from the `RUBYLIB` environment
35    /// variable.
36    ///
37    /// The `RUBYLIB` environment variable is read only once at the time this
38    /// method is called. The resolved load path is immutable.
39    ///
40    /// This method returns [`None`] if there are errors resolving the
41    /// `RUBYLIB` environment variable, if the `RUBYLIB` environment variable is
42    /// not set, or if the given `RUBYLIB` environment variable only contains
43    /// empty paths.
44    ///
45    /// # Examples
46    ///
47    /// ```no_run
48    /// use mezzaluna_load_path::RubyCore;
49    ///
50    /// let loader = RubyCore::new();
51    /// ```
52    #[inline]
53    #[must_use]
54    pub const fn new() -> Self {
55        Self { _private: () }
56    }
57
58    /// Return a reference to the load path for sources in the Ruby Core
59    /// library.
60    ///
61    /// Features in Ruby Core have the lowest priority, so the returned path
62    /// should appear last in `$LOAD_PATH`.
63    ///
64    /// # Examples
65    ///
66    /// ```
67    /// use std::path::Path;
68    /// use mezzaluna_load_path::RubyCore;
69    ///
70    /// let loader = RubyCore::new();
71    ///
72    /// if cfg!(windows) {
73    ///     assert_eq!(
74    ///         loader.core_load_path(),
75    ///         Path::new("c:/artichoke/virtual_root/site/core/lib"),
76    ///     );
77    /// } else {
78    ///     assert_eq!(
79    ///         loader.core_load_path(),
80    ///         Path::new("/artichoke/virtual_root/site/core/lib"),
81    ///     );
82    /// }
83    /// ```
84    #[inline]
85    #[must_use]
86    pub fn core_load_path(self) -> &'static Path {
87        if cfg!(windows) {
88            Path::new("c:/artichoke/virtual_root/site/core/lib")
89        } else {
90            Path::new("/artichoke/virtual_root/site/core/lib")
91        }
92    }
93
94    /// Return a reference to the load path for sources in the Ruby Standard
95    /// Library.
96    ///
97    /// Features in Ruby Standard Library have low priority, so the returned
98    /// path should appear second to last in `$LOAD_PATH` (only ahead of the
99    /// [core load path]).
100    ///
101    /// # Examples
102    ///
103    /// ```
104    /// use std::path::Path;
105    /// use mezzaluna_load_path::RubyCore;
106    ///
107    /// let loader = RubyCore::new();
108    ///
109    /// if cfg!(windows) {
110    ///     assert_eq!(
111    ///         loader.stdlib_load_path(),
112    ///         Path::new("c:/artichoke/virtual_root/site/stdlib/lib"),
113    ///     );
114    /// } else {
115    ///     assert_eq!(
116    ///         loader.stdlib_load_path(),
117    ///         Path::new("/artichoke/virtual_root/site/stdlib/lib"),
118    ///     );
119    /// }
120    /// ```
121    ///
122    /// [core load path]: Self::core_load_path
123    #[inline]
124    #[must_use]
125    pub fn stdlib_load_path(self) -> &'static Path {
126        if cfg!(windows) {
127            Path::new("c:/artichoke/virtual_root/site/stdlib/lib")
128        } else {
129            Path::new("/artichoke/virtual_root/site/stdlib/lib")
130        }
131    }
132
133    /// Return a reference to the paths in `$LOAD_PATH` parsed by this builder.
134    ///
135    /// Because the site paths have the lowest priority when loading
136    /// features, the returned paths should appear last in `$LOAD_PATH`.
137    ///
138    /// # Examples
139    ///
140    /// ```
141    /// use mezzaluna_load_path::RubyCore;
142    ///
143    /// let loader = RubyCore::new();
144    /// assert_eq!(
145    ///     loader.load_path(),
146    ///     [loader.stdlib_load_path(), loader.core_load_path()],
147    /// );
148    /// ```
149    #[inline]
150    #[must_use]
151    pub fn load_path(self) -> [&'static Path; 2] {
152        [self.stdlib_load_path(), self.core_load_path()]
153    }
154}
155
156#[cfg(test)]
157mod tests {
158    use std::path::Path;
159
160    use super::*;
161
162    #[test]
163    #[allow(dead_code)]
164    fn test_apis_are_const() {
165        const LOADER: RubyCore = RubyCore::new();
166    }
167
168    #[test]
169    fn test_load_path_has_two_entries() {
170        let loader = RubyCore::new();
171        assert_eq!(loader.load_path().len(), 2);
172    }
173
174    #[test]
175    fn test_load_path_has_stdlib_first() {
176        let loader = RubyCore::new();
177        assert_eq!(loader.load_path().first().copied().unwrap(), loader.stdlib_load_path());
178    }
179
180    #[test]
181    fn test_load_path_has_core_last() {
182        let loader = RubyCore::new();
183        assert_eq!(loader.load_path().last().copied().unwrap(), loader.core_load_path());
184    }
185
186    #[test]
187    fn test_all_paths_are_non_empty() {
188        let loader = RubyCore::new();
189        assert!(!loader.core_load_path().as_os_str().is_empty());
190        assert!(!loader.stdlib_load_path().as_os_str().is_empty());
191        assert!(loader.load_path().iter().copied().all(|p| !p.as_os_str().is_empty()));
192    }
193
194    #[test]
195    fn test_all_paths_are_absolute() {
196        let loader = RubyCore::new();
197        assert!(loader.core_load_path().is_absolute());
198        assert!(loader.stdlib_load_path().is_absolute());
199        assert!(loader.load_path().iter().copied().all(Path::is_absolute));
200    }
201}