xref: /freebsd/contrib/kyua/utils/fs/lua_module.cpp (revision b0d29bc47dba79f6f38e67eabadfb4b32ffd9390)
1*b0d29bc4SBrooks Davis // Copyright 2011 The Kyua Authors.
2*b0d29bc4SBrooks Davis // All rights reserved.
3*b0d29bc4SBrooks Davis //
4*b0d29bc4SBrooks Davis // Redistribution and use in source and binary forms, with or without
5*b0d29bc4SBrooks Davis // modification, are permitted provided that the following conditions are
6*b0d29bc4SBrooks Davis // met:
7*b0d29bc4SBrooks Davis //
8*b0d29bc4SBrooks Davis // * Redistributions of source code must retain the above copyright
9*b0d29bc4SBrooks Davis //   notice, this list of conditions and the following disclaimer.
10*b0d29bc4SBrooks Davis // * Redistributions in binary form must reproduce the above copyright
11*b0d29bc4SBrooks Davis //   notice, this list of conditions and the following disclaimer in the
12*b0d29bc4SBrooks Davis //   documentation and/or other materials provided with the distribution.
13*b0d29bc4SBrooks Davis // * Neither the name of Google Inc. nor the names of its contributors
14*b0d29bc4SBrooks Davis //   may be used to endorse or promote products derived from this software
15*b0d29bc4SBrooks Davis //   without specific prior written permission.
16*b0d29bc4SBrooks Davis //
17*b0d29bc4SBrooks Davis // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18*b0d29bc4SBrooks Davis // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19*b0d29bc4SBrooks Davis // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20*b0d29bc4SBrooks Davis // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21*b0d29bc4SBrooks Davis // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22*b0d29bc4SBrooks Davis // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23*b0d29bc4SBrooks Davis // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*b0d29bc4SBrooks Davis // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*b0d29bc4SBrooks Davis // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*b0d29bc4SBrooks Davis // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27*b0d29bc4SBrooks Davis // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*b0d29bc4SBrooks Davis 
29*b0d29bc4SBrooks Davis #include "utils/fs/lua_module.hpp"
30*b0d29bc4SBrooks Davis 
31*b0d29bc4SBrooks Davis extern "C" {
32*b0d29bc4SBrooks Davis #include <dirent.h>
33*b0d29bc4SBrooks Davis }
34*b0d29bc4SBrooks Davis 
35*b0d29bc4SBrooks Davis #include <cerrno>
36*b0d29bc4SBrooks Davis #include <cstring>
37*b0d29bc4SBrooks Davis #include <stdexcept>
38*b0d29bc4SBrooks Davis #include <string>
39*b0d29bc4SBrooks Davis 
40*b0d29bc4SBrooks Davis #include <lutok/operations.hpp>
41*b0d29bc4SBrooks Davis #include <lutok/stack_cleaner.hpp>
42*b0d29bc4SBrooks Davis #include <lutok/state.ipp>
43*b0d29bc4SBrooks Davis 
44*b0d29bc4SBrooks Davis #include "utils/format/macros.hpp"
45*b0d29bc4SBrooks Davis #include "utils/fs/operations.hpp"
46*b0d29bc4SBrooks Davis #include "utils/fs/path.hpp"
47*b0d29bc4SBrooks Davis #include "utils/sanity.hpp"
48*b0d29bc4SBrooks Davis 
49*b0d29bc4SBrooks Davis namespace fs = utils::fs;
50*b0d29bc4SBrooks Davis 
51*b0d29bc4SBrooks Davis 
52*b0d29bc4SBrooks Davis namespace {
53*b0d29bc4SBrooks Davis 
54*b0d29bc4SBrooks Davis 
55*b0d29bc4SBrooks Davis /// Given a path, qualifies it with the module's start directory if necessary.
56*b0d29bc4SBrooks Davis ///
57*b0d29bc4SBrooks Davis /// \param state The Lua state.
58*b0d29bc4SBrooks Davis /// \param path The path to qualify.
59*b0d29bc4SBrooks Davis ///
60*b0d29bc4SBrooks Davis /// \return The original path if it was absolute; otherwise the original path
61*b0d29bc4SBrooks Davis /// appended to the module's start directory.
62*b0d29bc4SBrooks Davis ///
63*b0d29bc4SBrooks Davis /// \throw std::runtime_error If the module's state has been corrupted.
64*b0d29bc4SBrooks Davis static fs::path
qualify_path(lutok::state & state,const fs::path & path)65*b0d29bc4SBrooks Davis qualify_path(lutok::state& state, const fs::path& path)
66*b0d29bc4SBrooks Davis {
67*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
68*b0d29bc4SBrooks Davis 
69*b0d29bc4SBrooks Davis     if (path.is_absolute()) {
70*b0d29bc4SBrooks Davis         return path;
71*b0d29bc4SBrooks Davis     } else {
72*b0d29bc4SBrooks Davis         state.get_global("_fs_start_dir");
73*b0d29bc4SBrooks Davis         if (!state.is_string(-1))
74*b0d29bc4SBrooks Davis             throw std::runtime_error("Missing _fs_start_dir global variable; "
75*b0d29bc4SBrooks Davis                                      "state corrupted?");
76*b0d29bc4SBrooks Davis         return fs::path(state.to_string(-1)) / path;
77*b0d29bc4SBrooks Davis     }
78*b0d29bc4SBrooks Davis }
79*b0d29bc4SBrooks Davis 
80*b0d29bc4SBrooks Davis 
81*b0d29bc4SBrooks Davis /// Safely gets a path from the Lua state.
82*b0d29bc4SBrooks Davis ///
83*b0d29bc4SBrooks Davis /// \param state The Lua state.
84*b0d29bc4SBrooks Davis /// \param index The position in the Lua stack that contains the path to query.
85*b0d29bc4SBrooks Davis ///
86*b0d29bc4SBrooks Davis /// \return The queried path.
87*b0d29bc4SBrooks Davis ///
88*b0d29bc4SBrooks Davis /// \throw fs::error If the value is not a valid path.
89*b0d29bc4SBrooks Davis /// \throw std::runtime_error If the value on the Lua stack is not convertible
90*b0d29bc4SBrooks Davis ///     to a path.
91*b0d29bc4SBrooks Davis static fs::path
to_path(lutok::state & state,const int index)92*b0d29bc4SBrooks Davis to_path(lutok::state& state, const int index)
93*b0d29bc4SBrooks Davis {
94*b0d29bc4SBrooks Davis     if (!state.is_string(index))
95*b0d29bc4SBrooks Davis         throw std::runtime_error("Need a string parameter");
96*b0d29bc4SBrooks Davis     return fs::path(state.to_string(index));
97*b0d29bc4SBrooks Davis }
98*b0d29bc4SBrooks Davis 
99*b0d29bc4SBrooks Davis 
100*b0d29bc4SBrooks Davis /// Lua binding for fs::path::basename.
101*b0d29bc4SBrooks Davis ///
102*b0d29bc4SBrooks Davis /// \pre stack(-1) The input path.
103*b0d29bc4SBrooks Davis /// \post stack(-1) The basename of the input path.
104*b0d29bc4SBrooks Davis ///
105*b0d29bc4SBrooks Davis /// \param state The Lua state.
106*b0d29bc4SBrooks Davis ///
107*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 1.
108*b0d29bc4SBrooks Davis static int
lua_fs_basename(lutok::state & state)109*b0d29bc4SBrooks Davis lua_fs_basename(lutok::state& state)
110*b0d29bc4SBrooks Davis {
111*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
112*b0d29bc4SBrooks Davis 
113*b0d29bc4SBrooks Davis     const fs::path path = to_path(state, -1);
114*b0d29bc4SBrooks Davis     state.push_string(path.leaf_name().c_str());
115*b0d29bc4SBrooks Davis     cleaner.forget();
116*b0d29bc4SBrooks Davis     return 1;
117*b0d29bc4SBrooks Davis }
118*b0d29bc4SBrooks Davis 
119*b0d29bc4SBrooks Davis 
120*b0d29bc4SBrooks Davis /// Lua binding for fs::path::dirname.
121*b0d29bc4SBrooks Davis ///
122*b0d29bc4SBrooks Davis /// \pre stack(-1) The input path.
123*b0d29bc4SBrooks Davis /// \post stack(-1) The directory part of the input path.
124*b0d29bc4SBrooks Davis ///
125*b0d29bc4SBrooks Davis /// \param state The Lua state.
126*b0d29bc4SBrooks Davis ///
127*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 1.
128*b0d29bc4SBrooks Davis static int
lua_fs_dirname(lutok::state & state)129*b0d29bc4SBrooks Davis lua_fs_dirname(lutok::state& state)
130*b0d29bc4SBrooks Davis {
131*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
132*b0d29bc4SBrooks Davis 
133*b0d29bc4SBrooks Davis     const fs::path path = to_path(state, -1);
134*b0d29bc4SBrooks Davis     state.push_string(path.branch_path().c_str());
135*b0d29bc4SBrooks Davis     cleaner.forget();
136*b0d29bc4SBrooks Davis     return 1;
137*b0d29bc4SBrooks Davis }
138*b0d29bc4SBrooks Davis 
139*b0d29bc4SBrooks Davis 
140*b0d29bc4SBrooks Davis /// Lua binding for fs::path::exists.
141*b0d29bc4SBrooks Davis ///
142*b0d29bc4SBrooks Davis /// \pre stack(-1) The input path.
143*b0d29bc4SBrooks Davis /// \post stack(-1) Whether the input path exists or not.
144*b0d29bc4SBrooks Davis ///
145*b0d29bc4SBrooks Davis /// \param state The Lua state.
146*b0d29bc4SBrooks Davis ///
147*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 1.
148*b0d29bc4SBrooks Davis static int
lua_fs_exists(lutok::state & state)149*b0d29bc4SBrooks Davis lua_fs_exists(lutok::state& state)
150*b0d29bc4SBrooks Davis {
151*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
152*b0d29bc4SBrooks Davis 
153*b0d29bc4SBrooks Davis     const fs::path path = qualify_path(state, to_path(state, -1));
154*b0d29bc4SBrooks Davis     state.push_boolean(fs::exists(path));
155*b0d29bc4SBrooks Davis     cleaner.forget();
156*b0d29bc4SBrooks Davis     return 1;
157*b0d29bc4SBrooks Davis }
158*b0d29bc4SBrooks Davis 
159*b0d29bc4SBrooks Davis 
160*b0d29bc4SBrooks Davis /// Lua binding for the files iterator.
161*b0d29bc4SBrooks Davis ///
162*b0d29bc4SBrooks Davis /// This function takes an open directory from the closure of the iterator and
163*b0d29bc4SBrooks Davis /// returns the next entry.  See lua_fs_files() for the iterator generator
164*b0d29bc4SBrooks Davis /// function.
165*b0d29bc4SBrooks Davis ///
166*b0d29bc4SBrooks Davis /// \pre upvalue(1) The userdata containing an open DIR* object.
167*b0d29bc4SBrooks Davis ///
168*b0d29bc4SBrooks Davis /// \param state The lua state.
169*b0d29bc4SBrooks Davis ///
170*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 0 if there are no more entries or
171*b0d29bc4SBrooks Davis /// 1 if an entry has been read.
172*b0d29bc4SBrooks Davis static int
files_iterator(lutok::state & state)173*b0d29bc4SBrooks Davis files_iterator(lutok::state& state)
174*b0d29bc4SBrooks Davis {
175*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
176*b0d29bc4SBrooks Davis 
177*b0d29bc4SBrooks Davis     DIR** dirp = state.to_userdata< DIR* >(state.upvalue_index(1));
178*b0d29bc4SBrooks Davis     const struct dirent* entry = ::readdir(*dirp);
179*b0d29bc4SBrooks Davis     if (entry == NULL)
180*b0d29bc4SBrooks Davis         return 0;
181*b0d29bc4SBrooks Davis     else {
182*b0d29bc4SBrooks Davis         state.push_string(entry->d_name);
183*b0d29bc4SBrooks Davis         cleaner.forget();
184*b0d29bc4SBrooks Davis         return 1;
185*b0d29bc4SBrooks Davis     }
186*b0d29bc4SBrooks Davis }
187*b0d29bc4SBrooks Davis 
188*b0d29bc4SBrooks Davis 
189*b0d29bc4SBrooks Davis /// Lua binding for the destruction of the files iterator.
190*b0d29bc4SBrooks Davis ///
191*b0d29bc4SBrooks Davis /// This function takes an open directory and closes it.  See lua_fs_files() for
192*b0d29bc4SBrooks Davis /// the iterator generator function.
193*b0d29bc4SBrooks Davis ///
194*b0d29bc4SBrooks Davis /// \pre stack(-1) The userdata containing an open DIR* object.
195*b0d29bc4SBrooks Davis /// \post The DIR* object is closed.
196*b0d29bc4SBrooks Davis ///
197*b0d29bc4SBrooks Davis /// \param state The lua state.
198*b0d29bc4SBrooks Davis ///
199*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 0.
200*b0d29bc4SBrooks Davis static int
files_gc(lutok::state & state)201*b0d29bc4SBrooks Davis files_gc(lutok::state& state)
202*b0d29bc4SBrooks Davis {
203*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
204*b0d29bc4SBrooks Davis 
205*b0d29bc4SBrooks Davis     PRE(state.is_userdata(-1));
206*b0d29bc4SBrooks Davis 
207*b0d29bc4SBrooks Davis     DIR** dirp = state.to_userdata< DIR* >(-1);
208*b0d29bc4SBrooks Davis     // For some reason, this may be called more than once.  I don't know why
209*b0d29bc4SBrooks Davis     // this happens, but we must protect against it.
210*b0d29bc4SBrooks Davis     if (*dirp != NULL) {
211*b0d29bc4SBrooks Davis         ::closedir(*dirp);
212*b0d29bc4SBrooks Davis         *dirp = NULL;
213*b0d29bc4SBrooks Davis     }
214*b0d29bc4SBrooks Davis 
215*b0d29bc4SBrooks Davis     return 0;
216*b0d29bc4SBrooks Davis }
217*b0d29bc4SBrooks Davis 
218*b0d29bc4SBrooks Davis 
219*b0d29bc4SBrooks Davis /// Lua binding to create an iterator to scan the contents of a directory.
220*b0d29bc4SBrooks Davis ///
221*b0d29bc4SBrooks Davis /// \pre stack(-1) The input path.
222*b0d29bc4SBrooks Davis /// \post stack(-1) The iterator function.
223*b0d29bc4SBrooks Davis ///
224*b0d29bc4SBrooks Davis /// \param state The Lua state.
225*b0d29bc4SBrooks Davis ///
226*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 1.
227*b0d29bc4SBrooks Davis static int
lua_fs_files(lutok::state & state)228*b0d29bc4SBrooks Davis lua_fs_files(lutok::state& state)
229*b0d29bc4SBrooks Davis {
230*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
231*b0d29bc4SBrooks Davis 
232*b0d29bc4SBrooks Davis     const fs::path path = qualify_path(state, to_path(state, -1));
233*b0d29bc4SBrooks Davis 
234*b0d29bc4SBrooks Davis     DIR** dirp = state.new_userdata< DIR* >();
235*b0d29bc4SBrooks Davis 
236*b0d29bc4SBrooks Davis     state.new_table();
237*b0d29bc4SBrooks Davis     state.push_string("__gc");
238*b0d29bc4SBrooks Davis     state.push_cxx_function(files_gc);
239*b0d29bc4SBrooks Davis     state.set_table(-3);
240*b0d29bc4SBrooks Davis 
241*b0d29bc4SBrooks Davis     state.set_metatable(-2);
242*b0d29bc4SBrooks Davis 
243*b0d29bc4SBrooks Davis     *dirp = ::opendir(path.c_str());
244*b0d29bc4SBrooks Davis     if (*dirp == NULL) {
245*b0d29bc4SBrooks Davis         const int original_errno = errno;
246*b0d29bc4SBrooks Davis         throw std::runtime_error(F("Failed to open directory: %s") %
247*b0d29bc4SBrooks Davis                                  std::strerror(original_errno));
248*b0d29bc4SBrooks Davis     }
249*b0d29bc4SBrooks Davis 
250*b0d29bc4SBrooks Davis     state.push_cxx_closure(files_iterator, 1);
251*b0d29bc4SBrooks Davis 
252*b0d29bc4SBrooks Davis     cleaner.forget();
253*b0d29bc4SBrooks Davis     return 1;
254*b0d29bc4SBrooks Davis }
255*b0d29bc4SBrooks Davis 
256*b0d29bc4SBrooks Davis 
257*b0d29bc4SBrooks Davis /// Lua binding for fs::path::is_absolute.
258*b0d29bc4SBrooks Davis ///
259*b0d29bc4SBrooks Davis /// \pre stack(-1) The input path.
260*b0d29bc4SBrooks Davis /// \post stack(-1) Whether the input path is absolute or not.
261*b0d29bc4SBrooks Davis ///
262*b0d29bc4SBrooks Davis /// \param state The Lua state.
263*b0d29bc4SBrooks Davis ///
264*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 1.
265*b0d29bc4SBrooks Davis static int
lua_fs_is_absolute(lutok::state & state)266*b0d29bc4SBrooks Davis lua_fs_is_absolute(lutok::state& state)
267*b0d29bc4SBrooks Davis {
268*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
269*b0d29bc4SBrooks Davis 
270*b0d29bc4SBrooks Davis     const fs::path path = to_path(state, -1);
271*b0d29bc4SBrooks Davis 
272*b0d29bc4SBrooks Davis     state.push_boolean(path.is_absolute());
273*b0d29bc4SBrooks Davis     cleaner.forget();
274*b0d29bc4SBrooks Davis     return 1;
275*b0d29bc4SBrooks Davis }
276*b0d29bc4SBrooks Davis 
277*b0d29bc4SBrooks Davis 
278*b0d29bc4SBrooks Davis /// Lua binding for fs::path::operator/.
279*b0d29bc4SBrooks Davis ///
280*b0d29bc4SBrooks Davis /// \pre stack(-2) The first input path.
281*b0d29bc4SBrooks Davis /// \pre stack(-1) The second input path.
282*b0d29bc4SBrooks Davis /// \post stack(-1) The concatenation of the two paths.
283*b0d29bc4SBrooks Davis ///
284*b0d29bc4SBrooks Davis /// \param state The Lua state.
285*b0d29bc4SBrooks Davis ///
286*b0d29bc4SBrooks Davis /// \return The number of result values, i.e. 1.
287*b0d29bc4SBrooks Davis static int
lua_fs_join(lutok::state & state)288*b0d29bc4SBrooks Davis lua_fs_join(lutok::state& state)
289*b0d29bc4SBrooks Davis {
290*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(state);
291*b0d29bc4SBrooks Davis 
292*b0d29bc4SBrooks Davis     const fs::path path1 = to_path(state, -2);
293*b0d29bc4SBrooks Davis     const fs::path path2 = to_path(state, -1);
294*b0d29bc4SBrooks Davis     state.push_string((path1 / path2).c_str());
295*b0d29bc4SBrooks Davis     cleaner.forget();
296*b0d29bc4SBrooks Davis     return 1;
297*b0d29bc4SBrooks Davis }
298*b0d29bc4SBrooks Davis 
299*b0d29bc4SBrooks Davis 
300*b0d29bc4SBrooks Davis }  // anonymous namespace
301*b0d29bc4SBrooks Davis 
302*b0d29bc4SBrooks Davis 
303*b0d29bc4SBrooks Davis /// Creates a Lua 'fs' module with a default start directory of ".".
304*b0d29bc4SBrooks Davis ///
305*b0d29bc4SBrooks Davis /// \post The global 'fs' symbol is set to a table that contains functions to a
306*b0d29bc4SBrooks Davis /// variety of utilites from the fs C++ module.
307*b0d29bc4SBrooks Davis ///
308*b0d29bc4SBrooks Davis /// \param s The Lua state.
309*b0d29bc4SBrooks Davis void
open_fs(lutok::state & s)310*b0d29bc4SBrooks Davis fs::open_fs(lutok::state& s)
311*b0d29bc4SBrooks Davis {
312*b0d29bc4SBrooks Davis     open_fs(s, fs::current_path());
313*b0d29bc4SBrooks Davis }
314*b0d29bc4SBrooks Davis 
315*b0d29bc4SBrooks Davis 
316*b0d29bc4SBrooks Davis /// Creates a Lua 'fs' module with an explicit start directory.
317*b0d29bc4SBrooks Davis ///
318*b0d29bc4SBrooks Davis /// \post The global 'fs' symbol is set to a table that contains functions to a
319*b0d29bc4SBrooks Davis /// variety of utilites from the fs C++ module.
320*b0d29bc4SBrooks Davis ///
321*b0d29bc4SBrooks Davis /// \param s The Lua state.
322*b0d29bc4SBrooks Davis /// \param start_dir The start directory to use in all operations that reference
323*b0d29bc4SBrooks Davis ///     the underlying file sytem.
324*b0d29bc4SBrooks Davis void
open_fs(lutok::state & s,const fs::path & start_dir)325*b0d29bc4SBrooks Davis fs::open_fs(lutok::state& s, const fs::path& start_dir)
326*b0d29bc4SBrooks Davis {
327*b0d29bc4SBrooks Davis     lutok::stack_cleaner cleaner(s);
328*b0d29bc4SBrooks Davis 
329*b0d29bc4SBrooks Davis     s.push_string(start_dir.str());
330*b0d29bc4SBrooks Davis     s.set_global("_fs_start_dir");
331*b0d29bc4SBrooks Davis 
332*b0d29bc4SBrooks Davis     std::map< std::string, lutok::cxx_function > members;
333*b0d29bc4SBrooks Davis     members["basename"] = lua_fs_basename;
334*b0d29bc4SBrooks Davis     members["dirname"] = lua_fs_dirname;
335*b0d29bc4SBrooks Davis     members["exists"] = lua_fs_exists;
336*b0d29bc4SBrooks Davis     members["files"] = lua_fs_files;
337*b0d29bc4SBrooks Davis     members["is_absolute"] = lua_fs_is_absolute;
338*b0d29bc4SBrooks Davis     members["join"] = lua_fs_join;
339*b0d29bc4SBrooks Davis     lutok::create_module(s, "fs", members);
340*b0d29bc4SBrooks Davis }
341