| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
 | //! @file utils.h
#ifndef STASIS_UTILS_H
#define STASIS_UTILS_H
#include <stdio.h>
#include <stdlib.h>
#include <dirent.h>
#include <string.h>
#include <unistd.h>
#include <limits.h>
#include <errno.h>
#include "system.h"
#if defined(STASIS_OS_WINDOWS)
#define PATH_ENV_VAR "path"
#define DIR_SEP "\\"
#define PATH_SEP ";"
#define LINE_SEP "\r\n"
#else
#define PATH_ENV_VAR "PATH"
#define DIR_SEP "/"
#define PATH_SEP ":"
#define LINE_SEP "\n"
#endif
#define STASIS_XML_PRETTY_PRINT_PROG "xmllint"
#define STASIS_XML_PRETTY_PRINT_ARGS "--format"
typedef int (ReaderFn)(size_t line, char **);
/**
 * Change directory. Push path on directory stack.
 *
 * ```c
 * pushd("/somepath");
 *
 * FILE fp = fopen("somefile", "w"); // i.e. /somepath/somefile
 * fprintf(fp, "Hello world.\n");
 * fclose(fp);
 *
 * popd();
 * ```
 *
 * @param path of directory
 * @return 0 on success, -1 on error
 */
int pushd(const char *path);
/**
 * Return from directory. Pop last path from directory stack.
 *
 * @see pushd
 * @return 0 on success, -1 if stack is empty
 */
int popd(void);
/**
 * Expand "~" to the user's home directory
 *
 * ```c
 * char *home = expandpath("~");             // == /home/username
 * char *config = expandpath("~/.config");   // == /home/username/.config
 * char *nope = expandpath("/tmp/test");     // == /tmp/test
 * char *nada = expandpath("/~/broken");     // == /~/broken
 *
 * free(home);
 * free(config);
 * free(nope);
 * free(nada);
 * ```
 *
 * @param _path (Must start with a `~`)
 * @return success=expanded path or original path, failure=NULL
 */
char *expandpath(const char *_path);
/**
 * Remove a directory tree recursively
 *
 * ```c
 * mkdirs("a/b/c");
 * rmtree("a");
 * // a/b/c is removed
 * ```
 *
 * @param _path
 * @return 0 on success, -1 on error
 */
int rmtree(char *_path);
char **file_readlines(const char *filename, size_t start, size_t limit, ReaderFn *readerFn);
/**
 * Strip directory from file name
 * Note: Caller is responsible for freeing memory
 *
 * @param _path
 * @return success=file name, failure=NULL
 */
char *path_basename(char *path);
/**
 * Return parent directory of file, or the parent of a directory
 *
 * @param path
 * @return success=directory, failure=empty string
 */
char *path_dirname(char *path);
/**
 * Scan PATH directories for a named program
 * @param name program name
 * @return path to program, or NULL on error
 */
char *find_program(const char *name);
/**
 * Create an empty file, or update modified timestamp on an existing file
 * @param filename file to touch
 * @return 0 on success, 1 on error
 */
int touch(const char *filename);
/**
 * Clone a git repository
 *
 * ```c
 * struct Process proc;
 * memset(proc, 0, sizeof(proc));
 *
 * if (git_clone(&proc, "https://github.com/myuser/myrepo", "./repos", "unstable_branch")) {
 *     fprintf(stderr, "Failed to clone repository\n");
 *     exit(1);
 * }
 *
 * if (pushd("./repos/myrepo")) {
 *     fprintf(stderr, "Unable to enter repository directory\n");
 * } else {
 *     // do something with repository
 *     popd();
 * }
 * ```
 *
 * @see pushd
 *
 * @param proc Process struct
 * @param url URL (or file system path) of repoistory to clone
 * @param destdir destination directory
 * @param gitref commit/branch/tag of checkout (NULL will use HEAD of default branch for repo)
 * @return exit code from "git"
 */
int git_clone(struct Process *proc, char *url, char *destdir, char *gitref);
/**
 * Git describe wrapper
 * @param path to repository
 * @return output from "git describe", or NULL on error
 */
char *git_describe(const char *path);
/**
 * Git rev-parse wrapper
 * @param path to repository
 * @param args to pass to git rev-parse
 * @return output from "git rev-parse", or NULL on error
 */
char *git_rev_parse(const char *path, char *args);
/**
 * Helper function to initialize simple STASIS internal path strings
 *
 * ```c
 * char *mypath = NULL;
 *
 * if (path_store(&mypath, PATH_MAX, "/some", "path")) {
 *     fprintf(stderr, "Unable to allocate memory for path elements\n");
 *     exit(1);
 * }
 * // mypath is allocated to size PATH_MAX and contains the string: /some/path
 * // base+path will truncate at maxlen - 1
 * ```
 *
 * @param destptr address of destination string pointer
 * @param maxlen maximum length of the path
 * @param base path
 * @param path to append to base
 * @return 0 on success, -1 on error
 */
int path_store(char **destptr, size_t maxlen, const char *base, const char *path);
#if defined(STASIS_DUMB_TERMINAL)
#define STASIS_COLOR_RED ""
#define STASIS_COLOR_GREEN ""
#define STASIS_COLOR_YELLOW ""
#define STASIS_COLOR_BLUE ""
#define STASIS_COLOR_WHITE ""
#define STASIS_COLOR_RESET ""
#else
//! Set output color to red
#define STASIS_COLOR_RED "\e[1;91m"
//! Set output color to green
#define STASIS_COLOR_GREEN "\e[1;92m"
//! Set output color to yellow
#define STASIS_COLOR_YELLOW "\e[1;93m"
//! Set output color to blue
#define STASIS_COLOR_BLUE "\e[1;94m"
//! Set output color to white
#define STASIS_COLOR_WHITE "\e[1;97m"
//! Reset output color to terminal default
#define STASIS_COLOR_RESET "\e[0;37m\e[0m"
#endif
#define STASIS_MSG_SUCCESS 0
//! Suppress printing of the message text
#define STASIS_MSG_NOP 1 << 0
//! The message is an error
#define STASIS_MSG_ERROR 1 << 1
//! The message is a warning
#define STASIS_MSG_WARN 1 << 2
//! The message will be indented once
#define STASIS_MSG_L1 1 << 3
//! The message will be indented twice
#define STASIS_MSG_L2 1 << 4
//! The message will be indented thrice
#define STASIS_MSG_L3 1 << 5
//! The message will only be printed in verbose mode
#define STASIS_MSG_RESTRICT 1 << 6
void msg(unsigned type, char *fmt, ...);
// Enter an interactive shell that ends the program on-exit
void debug_shell();
/**
 * Creates a temporary file returning an open file pointer via @a fp, and the
 * path to the file. The caller is responsible for closing @a fp and
 * free()ing the returned file path.
 *
 * ```c
 * FILE *fp = NULL;
 * char *tempfile = xmkstemp(&fp, "r+");
 * if (!fp || !tempfile) {
 *     fprintf(stderr, "Failed to generate temporary file for read/write\n");
 *     exit(1);
 * }
 * ```
 *
 * @param fp pointer to FILE (to be initialized)
 * @param mode fopen() style file mode string
 * @return system path to the temporary file
 * @return NULL on failure
 */
char *xmkstemp(FILE **fp, const char *mode);
/**
 * Is the path an empty directory structure?
 *
 * ```c
 * if (isempty_dir("/some/path")) {
 *     fprintf(stderr, "The directory is is empty!\n");
 * } else {
 *     printf("The directory contains dirs/files\n");
 * }
 * ```
 *
 * @param path directory
 * @return 0 = no, 1 = yes
 */
int isempty_dir(const char *path);
/**
 * Rewrite an XML file with a pretty printer command
 * @param filename path to modify
 * @param pretty_print_prog program to call
 * @param pretty_print_args arguments to pass to program
 * @return 0 on success, -1 on error
 */
int xml_pretty_print_in_place(const char *filename, const char *pretty_print_prog, const char *pretty_print_args);
/**
 * Applies STASIS fixups to a tox ini config
 * @param filename path to tox.ini
 * @param result path to processed configuration
 * @return 0 on success, -1 on error
 */
int fix_tox_conf(const char *filename, char **result);
char *collapse_whitespace(char **s);
/**
 * Write ***REDACTED*** in dest for each occurrence of to_redacted token present in src
 *
 * ```c
 * char command[PATH_MAX] = {0};
 * char command_redacted[PATH_MAX] = {0};
 * const char *password = "abc123";
 * const char *host = "myhostname";
 * const char *to_redact_case1[] = {password, host, NULL};
 * const char *to_redact_case2[] = {password, "--host", NULL};
 * const char *to_redact_case3[] = {password, "--host", host, NULL};
 *
 * sprintf(command, "echo %s | program --host=%s -", password, host);
 *
 * // CASE 1
 * redact_sensitive(to_redact_case1, command, command_redacted, sizeof(command_redacted) - 1);
 * printf("executing: %s\n", command_redacted);
 * // User sees:
 * // executing: echo ***REDACTED*** | program --host=***REDACTED*** -
 * system(command);
 *
 * // CASE 2 remove an entire argument
 * redact_sensitive(to_redact_case2, command, command_redacted, sizeof(command_redacted) - 1);
 * printf("executing: %s\n", command_redacted);
 * // User sees:
 * // executing: echo ***REDACTED*** | program ***REDACTED*** -
 * system(command);
 *
 * // CASE 3 remove it all (noisy)
 * redact_sensitive(to_redact_case3, command, command_redacted, sizeof(command_redacted) - 1);
 * printf("executing: %s\n", command_redacted);
 * // User sees:
 * // executing: echo ***REDACTED*** | program ***REDACTED***=***REDACTED*** -
 * system(command);
 * ```
 *
 * @param to_redact array of tokens to redact
 * @param src input string
 * @param dest output string
 * @param maxlen maximum length of dest byte array
 * @return 0 on success, -1 on error
 */
int redact_sensitive(const char **to_redact, size_t to_redact_size, char *src, char *dest, size_t maxlen);
/**
 * Given a directory path, return a list of files
 *
 * ~~~{.c}
 * struct StrList *files;
 *
 * basepath = ".";
 * files = listdir(basepath);
 * for (size_t i = 0; i < strlist_count(files); i++) {
 *     char *filename = strlist_item(files, i);
 *     printf("%s/%s\n", basepath, filename);
 * }
 * guard_strlist_free(&files);
 * ~~~
 *
 * @param path of a directory
 * @return a StrList containing file names
 */
struct StrList *listdir(const char *path);
/**
 * Get CPU count
 * @return CPU count on success, zero on error
 */
long get_cpu_count();
/**
 * Create all leafs in directory path
 * @param _path directory path to create
 * @param mode mode_t permissions
 * @return
 */
int mkdirs(const char *_path, mode_t mode);
/**
 * Return pointer to a (possible) version specifier
 *
 * ```c
 * char s[] = "abc==1.2.3";
 * char *spec_begin = find_version_spec(s);
 * // spec_begin is "==1.2.3"
 *
 * char package_name[255];
 * char s[] = "abc";
 * char *spec_pos = find_version_spec(s);
 * if (spec_pos) {
 *     strncpy(package_name, spec_pos - s);
 *     // use spec
 * } else {
 *     // spec not found
 * }
 *
 * @param str a pointer to a buffer containing a package spec (i.e. abc==1.2.3, abc>=1.2.3, abc)
 * @return a pointer to the first occurrence of a version spec character
 * @return NULL if not found
 */
char *find_version_spec(char *package_name);
#endif //STASIS_UTILS_H
 |