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