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}