2018-03-29 00:50:20 +00:00
|
|
|
/*
|
2019-01-06 08:23:03 +00:00
|
|
|
* Copyright (c) 2019 Calvin Rose
|
2018-03-29 00:50:20 +00:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to
|
|
|
|
* deal in the Software without restriction, including without limitation the
|
|
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
|
|
* IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2019-01-24 05:15:58 +00:00
|
|
|
#ifndef JANET_AMALG
|
2019-02-19 01:13:35 +00:00
|
|
|
#include <janet.h>
|
2019-01-24 05:15:58 +00:00
|
|
|
#include "util.h"
|
|
|
|
#endif
|
|
|
|
|
2018-03-29 00:50:20 +00:00
|
|
|
#include <stdlib.h>
|
2019-03-29 03:22:58 +00:00
|
|
|
|
|
|
|
#ifndef JANET_REDUCED_OS
|
|
|
|
|
2018-05-13 00:31:28 +00:00
|
|
|
#include <time.h>
|
2019-03-29 03:34:24 +00:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sys/stat.h>
|
2018-05-13 00:31:28 +00:00
|
|
|
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2019-03-29 03:34:24 +00:00
|
|
|
#include <windows.h>
|
2018-05-19 05:09:56 +00:00
|
|
|
#include <direct.h>
|
2019-03-29 03:34:24 +00:00
|
|
|
#include <sys/utime.h>
|
|
|
|
#include <io.h>
|
2019-05-30 23:14:54 +00:00
|
|
|
#include <process.h>
|
2018-05-13 00:31:28 +00:00
|
|
|
#else
|
2019-05-30 23:13:13 +00:00
|
|
|
#include <spawn.h>
|
2019-03-29 03:22:58 +00:00
|
|
|
#include <utime.h>
|
2018-05-13 00:31:28 +00:00
|
|
|
#include <unistd.h>
|
2019-03-29 03:22:58 +00:00
|
|
|
#include <dirent.h>
|
2018-07-07 01:50:59 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
2019-05-31 17:43:38 +00:00
|
|
|
extern char **environ;
|
2018-05-13 00:31:28 +00:00
|
|
|
#endif
|
2018-03-29 00:50:20 +00:00
|
|
|
|
2018-09-23 01:46:50 +00:00
|
|
|
/* For macos */
|
|
|
|
#ifdef __MACH__
|
|
|
|
#include <mach/clock.h>
|
|
|
|
#include <mach/mach.h>
|
|
|
|
#endif
|
|
|
|
|
2019-03-29 03:34:24 +00:00
|
|
|
#endif /* JANET_REDCUED_OS */
|
2019-03-29 03:22:58 +00:00
|
|
|
|
|
|
|
/* Core OS functions */
|
|
|
|
|
|
|
|
/* Full OS functions */
|
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_which(int32_t argc, Janet *argv) {
|
2019-01-06 01:45:24 +00:00
|
|
|
janet_fixarity(argc, 0);
|
2019-01-06 01:09:03 +00:00
|
|
|
(void) argv;
|
2019-02-20 01:51:34 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
return janet_ckeywordv("windows");
|
|
|
|
#elif __APPLE__
|
|
|
|
return janet_ckeywordv("macos");
|
|
|
|
#elif defined(__EMSCRIPTEN__)
|
|
|
|
return janet_ckeywordv("web");
|
|
|
|
#else
|
|
|
|
return janet_ckeywordv("posix");
|
|
|
|
#endif
|
2018-08-07 04:54:47 +00:00
|
|
|
}
|
|
|
|
|
2019-03-29 03:22:58 +00:00
|
|
|
static Janet os_exit(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 0, 1);
|
|
|
|
if (argc == 0) {
|
|
|
|
exit(EXIT_SUCCESS);
|
|
|
|
} else if (janet_checkint(argv[0])) {
|
|
|
|
exit(janet_unwrap_integer(argv[0]));
|
|
|
|
} else {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
return janet_wrap_nil();
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef JANET_REDUCED_OS
|
2019-05-30 22:40:10 +00:00
|
|
|
/* Provide a dud os/getenv so boot.janet and init.janet work, but nothing else */
|
2019-03-29 03:22:58 +00:00
|
|
|
|
|
|
|
static Janet os_getenv(int32_t argc, Janet *argv) {
|
|
|
|
(void) argv;
|
|
|
|
janet_fixarity(argc, 1);
|
|
|
|
return janet_wrap_nil();
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
/* Provide full os functionality */
|
|
|
|
|
2019-05-30 22:40:10 +00:00
|
|
|
#define JANET_OS_EFLAG_E 0x1
|
|
|
|
#define JANET_OS_EFLAG_P 0x2
|
|
|
|
|
|
|
|
/* Get flags */
|
|
|
|
/* Unfortunately, execvpe is linux (glibc) only. Instead, we can switch
|
|
|
|
* between the more portable execve, execvp, or execv.
|
|
|
|
* Use the :e or :p flag for execve and execvp respectively. Eventually
|
|
|
|
* :ep or :pe could be execvpe. */
|
|
|
|
static int os_execute_flags(int32_t argc, const Janet *argv) {
|
|
|
|
if (argc < 2) return 0;
|
|
|
|
int flags = 0;
|
|
|
|
if (argc > 1) {
|
|
|
|
const uint8_t *f = janet_getkeyword(argv, 1);
|
|
|
|
int32_t len = janet_string_length(f);
|
|
|
|
for (int32_t i = 0; i < len; i++) {
|
|
|
|
if (f[i] == 'e') flags |= JANET_OS_EFLAG_E;
|
|
|
|
if (f[i] == 'p') flags |= JANET_OS_EFLAG_P;
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-30 22:40:10 +00:00
|
|
|
return flags;
|
|
|
|
}
|
2018-07-07 01:50:59 +00:00
|
|
|
|
2019-05-30 22:40:10 +00:00
|
|
|
/* Get env for os_execute (execv family of functions, as well as CreateProcess) */
|
|
|
|
static char **os_execute_env(int32_t argc, const Janet *argv) {
|
|
|
|
char **envp = NULL;
|
|
|
|
if (argc > 2) {
|
|
|
|
JanetDictView dict = janet_getdictionary(argv, 2);
|
|
|
|
envp = malloc(sizeof(char *) * (dict.len + 1));
|
|
|
|
if (NULL == envp) {
|
|
|
|
JANET_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
int32_t j = 0;
|
|
|
|
for (int32_t i = 0; i < dict.cap; i++) {
|
|
|
|
const JanetKV *kv = dict.kvs + i;
|
|
|
|
if (!janet_checktype(kv->key, JANET_STRING)) continue;
|
|
|
|
if (!janet_checktype(kv->value, JANET_STRING)) continue;
|
|
|
|
const uint8_t *keys = janet_unwrap_string(kv->key);
|
|
|
|
const uint8_t *vals = janet_unwrap_string(kv->value);
|
|
|
|
int32_t klen = janet_string_length(keys);
|
|
|
|
int32_t vlen = janet_string_length(vals);
|
|
|
|
/* Check keys has no embedded 0s or =s. */
|
|
|
|
int skip = 0;
|
|
|
|
for (int32_t k = 0; k < klen; k++) {
|
|
|
|
if (keys[k] == '\0' || keys[k] == '=') {
|
|
|
|
skip = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (skip) continue;
|
|
|
|
char *envitem = malloc(klen + vlen + 2);
|
|
|
|
if (NULL == envitem) {
|
|
|
|
JANET_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
memcpy(envitem, keys, klen);
|
|
|
|
envitem[klen] = '=';
|
|
|
|
memcpy(envitem + klen + 1, vals, vlen);
|
|
|
|
envitem[klen + vlen + 1] = 0;
|
|
|
|
envp[j++] = envitem;
|
|
|
|
}
|
|
|
|
envp[j] = NULL;
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
2019-05-30 22:40:10 +00:00
|
|
|
return envp;
|
|
|
|
}
|
2018-07-07 01:50:59 +00:00
|
|
|
|
2019-05-30 22:40:10 +00:00
|
|
|
/* Free memory from os_execute */
|
|
|
|
static void os_execute_cleanup(char **envp, const char **child_argv) {
|
2019-05-31 17:43:38 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
(void) child_argv;
|
|
|
|
#else
|
2019-05-30 23:21:11 +00:00
|
|
|
free((void *)child_argv);
|
2019-05-31 17:43:38 +00:00
|
|
|
#endif
|
2019-05-30 22:40:10 +00:00
|
|
|
if (NULL != envp) {
|
|
|
|
char **envitem = envp;
|
|
|
|
while (*envitem != NULL) {
|
|
|
|
free(*envitem);
|
|
|
|
envitem++;
|
|
|
|
}
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
2019-05-30 22:40:10 +00:00
|
|
|
free(envp);
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
2019-05-30 22:40:10 +00:00
|
|
|
|
2019-05-31 17:43:38 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
/* Windows processes created via CreateProcess get only one command line argument string, and
|
|
|
|
* must parse this themselves. Each processes is free to do this however they like, but the
|
|
|
|
* standard parsing method is CommandLineToArgvW. We need to properly escape arguments into
|
|
|
|
* a single string of this format. Returns a buffer that can be cast into a c string. */
|
|
|
|
static JanetBuffer *os_exec_escape(JanetView args) {
|
|
|
|
JanetBuffer *b = janet_buffer(0);
|
|
|
|
for (int32_t i = 0; i < args.len; i++) {
|
|
|
|
const char *arg = janet_getcstring(args.items, i);
|
2019-05-30 22:40:10 +00:00
|
|
|
|
2019-05-31 17:43:38 +00:00
|
|
|
/* Push leading space if not first */
|
|
|
|
if (i) janet_buffer_push_u8(b, ' ');
|
|
|
|
|
|
|
|
/* Find first special character */
|
|
|
|
const char *first_spec = arg;
|
|
|
|
while (*first_spec) {
|
|
|
|
switch (*first_spec) {
|
|
|
|
case ' ':
|
|
|
|
case '\t':
|
|
|
|
case '\v':
|
|
|
|
case '\n':
|
|
|
|
case '"':
|
|
|
|
goto found;
|
|
|
|
case '\0':
|
|
|
|
janet_panic("embedded 0 not allowed in command line string");
|
|
|
|
default:
|
|
|
|
first_spec++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
found:
|
|
|
|
|
|
|
|
/* Check if needs escape */
|
|
|
|
if (*first_spec == '\0') {
|
|
|
|
/* No escape needed */
|
|
|
|
janet_buffer_push_cstring(b, arg);
|
|
|
|
} else {
|
|
|
|
/* Escape */
|
|
|
|
janet_buffer_push_u8(b, '"');
|
|
|
|
for (const char *c = arg; ; c++) {
|
|
|
|
unsigned numBackSlashes = 0;
|
|
|
|
while (*c == '\\') {
|
|
|
|
c++;
|
|
|
|
numBackSlashes++;
|
|
|
|
}
|
|
|
|
if (*c == '"') {
|
|
|
|
/* Escape all backslashes and double quote mark */
|
|
|
|
int32_t n = 2 * numBackSlashes + 1;
|
|
|
|
janet_buffer_extra(b, n + 1);
|
|
|
|
memset(b->data + b->count, '\\', n);
|
|
|
|
b->count += n;
|
|
|
|
janet_buffer_push_u8(b, '"');
|
|
|
|
} else if (*c) {
|
|
|
|
/* Don't escape backslashes. */
|
|
|
|
int32_t n = numBackSlashes;
|
|
|
|
janet_buffer_extra(b, n + 1);
|
|
|
|
memset(b->data + b->count, '\\', n);
|
|
|
|
b->count += n;
|
|
|
|
janet_buffer_push_u8(b, *c);
|
|
|
|
} else {
|
|
|
|
/* we finished Escape all backslashes */
|
|
|
|
int32_t n = 2 * numBackSlashes;
|
|
|
|
janet_buffer_extra(b, n + 1);
|
|
|
|
memset(b->data + b->count, '\\', n);
|
|
|
|
b->count += n;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
janet_buffer_push_u8(b, '"');
|
|
|
|
}
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
2019-05-31 17:43:38 +00:00
|
|
|
janet_buffer_push_u8(b, 0);
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static Janet os_execute(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 1, 3);
|
2019-05-30 22:40:10 +00:00
|
|
|
|
|
|
|
/* Get flags */
|
|
|
|
int flags = os_execute_flags(argc, argv);
|
|
|
|
|
|
|
|
/* Get environment */
|
|
|
|
char **envp = os_execute_env(argc, argv);
|
|
|
|
|
2019-05-31 17:43:38 +00:00
|
|
|
/* Get arguments */
|
|
|
|
JanetView exargs = janet_getindexed(argv, 0);
|
|
|
|
if (exargs.len < 1) {
|
|
|
|
janet_panic("expected at least 1 command line argument");
|
|
|
|
}
|
2019-05-30 22:40:10 +00:00
|
|
|
|
2019-05-31 17:45:39 +00:00
|
|
|
/* Result */
|
|
|
|
int status = 0;
|
|
|
|
|
2019-05-30 22:40:10 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
|
2019-05-31 17:43:38 +00:00
|
|
|
JanetBuffer *buf = os_exec_escape(exargs);
|
|
|
|
if (buf->count > 1025) {
|
|
|
|
janet_panic("command line string too long");
|
|
|
|
}
|
|
|
|
const char *path = (const char *) janet_unwrap_string(exargs.items[0]);
|
|
|
|
char *cargv[2] = {(char *) buf->data, NULL};
|
|
|
|
|
2019-05-30 22:40:10 +00:00
|
|
|
/* Use _spawn family of functions. */
|
|
|
|
/* Windows docs say do this before any spawns. */
|
|
|
|
_flushall();
|
|
|
|
|
|
|
|
if (flags & (JANET_OS_EFLAG_P | JANET_OS_EFLAG_E)) {
|
2019-05-31 17:43:38 +00:00
|
|
|
status = (int) _spawnvpe(_P_WAIT, path, cargv, envp);
|
2019-05-30 22:40:10 +00:00
|
|
|
} else if (flags & JANET_OS_EFLAG_P) {
|
2019-05-31 17:43:38 +00:00
|
|
|
status = (int) _spawnvp(_P_WAIT, path, cargv);
|
2019-05-30 22:40:10 +00:00
|
|
|
} else if (flags & JANET_OS_EFLAG_E) {
|
2019-05-31 17:43:38 +00:00
|
|
|
status = (int) _spawnve(_P_WAIT, path, cargv, envp);
|
2019-02-22 17:10:27 +00:00
|
|
|
} else {
|
2019-05-31 17:43:38 +00:00
|
|
|
status = (int) _spawnv(_P_WAIT, path, cargv);
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
2019-05-30 22:40:10 +00:00
|
|
|
|
2019-05-31 17:43:38 +00:00
|
|
|
os_execute_cleanup(envp, NULL);
|
2019-01-06 01:09:03 +00:00
|
|
|
return janet_wrap_integer(status);
|
2019-05-30 22:40:10 +00:00
|
|
|
#else
|
|
|
|
|
2019-05-31 17:43:38 +00:00
|
|
|
const char **child_argv = malloc(sizeof(char *) * (exargs.len + 1));
|
|
|
|
if (NULL == child_argv) {
|
|
|
|
JANET_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
for (int32_t i = 0; i < exargs.len; i++) {
|
|
|
|
child_argv[i] = janet_getcstring(exargs.items, i);
|
|
|
|
}
|
|
|
|
child_argv[exargs.len] = NULL;
|
|
|
|
/* Coerce to form that works for spawn. I'm fairly confident no implementation
|
|
|
|
* of posix_spawn would modify the argv array passed in. */
|
|
|
|
char *const *cargv = (char *const *)child_argv;
|
|
|
|
|
2019-05-30 22:40:10 +00:00
|
|
|
/* Use posix_spawn to spawn new process */
|
|
|
|
pid_t pid;
|
|
|
|
if (flags & JANET_OS_EFLAG_P) {
|
|
|
|
status = posix_spawnp(&pid,
|
|
|
|
child_argv[0], NULL, NULL, cargv,
|
2019-05-31 17:43:38 +00:00
|
|
|
(flags & JANET_OS_EFLAG_E) ? envp : environ);
|
2019-05-30 22:40:10 +00:00
|
|
|
} else {
|
|
|
|
status = posix_spawn(&pid,
|
|
|
|
child_argv[0], NULL, NULL, cargv,
|
2019-05-31 17:43:38 +00:00
|
|
|
(flags & JANET_OS_EFLAG_E) ? envp : environ);
|
2019-05-30 22:40:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for child */
|
|
|
|
if (status) {
|
|
|
|
os_execute_cleanup(envp, child_argv);
|
|
|
|
janet_panic(strerror(status));
|
|
|
|
} else {
|
|
|
|
waitpid(pid, &status, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
os_execute_cleanup(envp, child_argv);
|
|
|
|
return janet_wrap_integer(WEXITSTATUS(status));
|
2018-07-07 01:50:59 +00:00
|
|
|
#endif
|
2019-05-30 22:40:10 +00:00
|
|
|
}
|
2018-07-07 01:50:59 +00:00
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_shell(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 0, 1);
|
|
|
|
const char *cmd = argc
|
2019-03-29 03:22:58 +00:00
|
|
|
? janet_getcstring(argv, 0)
|
2019-02-20 01:51:34 +00:00
|
|
|
: NULL;
|
2018-03-29 00:50:20 +00:00
|
|
|
int stat = system(cmd);
|
2019-01-06 01:09:03 +00:00
|
|
|
return argc
|
2019-02-20 01:51:34 +00:00
|
|
|
? janet_wrap_integer(stat)
|
|
|
|
: janet_wrap_boolean(stat);
|
2018-03-29 00:50:20 +00:00
|
|
|
}
|
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_getenv(int32_t argc, Janet *argv) {
|
2019-01-06 01:45:24 +00:00
|
|
|
janet_fixarity(argc, 1);
|
2019-03-29 03:22:58 +00:00
|
|
|
const char *cstr = janet_getcstring(argv, 0);
|
2018-03-29 00:50:20 +00:00
|
|
|
const char *res = getenv(cstr);
|
2019-03-29 03:22:58 +00:00
|
|
|
return res
|
2019-02-20 01:51:34 +00:00
|
|
|
? janet_cstringv(res)
|
|
|
|
: janet_wrap_nil();
|
2018-03-29 00:50:20 +00:00
|
|
|
}
|
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_setenv(int32_t argc, Janet *argv) {
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2018-06-08 19:58:23 +00:00
|
|
|
#define SETENV(K,V) _putenv_s(K, V)
|
|
|
|
#define UNSETENV(K) _putenv_s(K, "")
|
2018-03-29 00:50:20 +00:00
|
|
|
#else
|
2018-06-08 19:58:23 +00:00
|
|
|
#define SETENV(K,V) setenv(K, V, 1)
|
|
|
|
#define UNSETENV(K) unsetenv(K)
|
|
|
|
#endif
|
2019-01-06 01:09:03 +00:00
|
|
|
janet_arity(argc, 1, 2);
|
2019-03-29 03:22:58 +00:00
|
|
|
const char *ks = janet_getcstring(argv, 0);
|
2019-01-06 01:09:03 +00:00
|
|
|
if (argc == 1 || janet_checktype(argv[1], JANET_NIL)) {
|
2018-06-08 19:58:23 +00:00
|
|
|
UNSETENV(ks);
|
2018-03-29 00:50:20 +00:00
|
|
|
} else {
|
2019-03-29 03:22:58 +00:00
|
|
|
SETENV(ks, janet_getcstring(argv, 1));
|
2018-05-13 00:31:28 +00:00
|
|
|
}
|
2019-01-06 01:09:03 +00:00
|
|
|
return janet_wrap_nil();
|
2018-05-13 00:31:28 +00:00
|
|
|
}
|
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_time(int32_t argc, Janet *argv) {
|
2019-01-06 01:45:24 +00:00
|
|
|
janet_fixarity(argc, 0);
|
2019-01-06 01:09:03 +00:00
|
|
|
(void) argv;
|
2018-09-23 01:46:50 +00:00
|
|
|
double dtime = (double)(time(NULL));
|
2019-01-06 01:09:03 +00:00
|
|
|
return janet_wrap_number(dtime);
|
2018-09-23 01:46:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Clock shims */
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2018-09-23 01:46:50 +00:00
|
|
|
static int gettime(struct timespec *spec) {
|
2018-07-09 01:10:15 +00:00
|
|
|
int64_t wintime = 0LL;
|
2019-02-20 01:51:34 +00:00
|
|
|
GetSystemTimeAsFileTime((FILETIME *)&wintime);
|
2018-07-09 01:10:15 +00:00
|
|
|
/* Windows epoch is January 1, 1601 apparently*/
|
|
|
|
wintime -= 116444736000000000LL;
|
2018-07-08 23:27:11 +00:00
|
|
|
spec->tv_sec = wintime / 10000000LL;
|
|
|
|
/* Resolution is 100 nanoseconds. */
|
|
|
|
spec->tv_nsec = wintime % 10000000LL * 100;
|
|
|
|
return 0;
|
|
|
|
}
|
2018-09-23 01:46:50 +00:00
|
|
|
#elif defined(__MACH__)
|
|
|
|
static int gettime(struct timespec *spec) {
|
|
|
|
clock_serv_t cclock;
|
|
|
|
mach_timespec_t mts;
|
|
|
|
host_get_clock_service(mach_host_self(), CALENDAR_CLOCK, &cclock);
|
|
|
|
clock_get_time(cclock, &mts);
|
|
|
|
mach_port_deallocate(mach_task_self(), cclock);
|
|
|
|
spec->tv_sec = mts.tv_sec;
|
|
|
|
spec->tv_nsec = mts.tv_nsec;
|
|
|
|
return 0;
|
2018-09-12 01:33:50 +00:00
|
|
|
}
|
2018-11-16 21:24:10 +00:00
|
|
|
#else
|
2018-09-23 01:46:50 +00:00
|
|
|
#define gettime(TV) clock_gettime(CLOCK_MONOTONIC, (TV))
|
|
|
|
#endif
|
2018-09-12 01:33:50 +00:00
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_clock(int32_t argc, Janet *argv) {
|
2019-01-06 01:45:24 +00:00
|
|
|
janet_fixarity(argc, 0);
|
2019-01-06 01:09:03 +00:00
|
|
|
(void) argv;
|
2018-07-08 23:27:11 +00:00
|
|
|
struct timespec tv;
|
2019-01-06 01:09:03 +00:00
|
|
|
if (gettime(&tv)) janet_panic("could not get time");
|
2018-07-08 23:27:11 +00:00
|
|
|
double dtime = tv.tv_sec + (tv.tv_nsec / 1E9);
|
2019-01-06 01:09:03 +00:00
|
|
|
return janet_wrap_number(dtime);
|
2018-05-13 00:31:28 +00:00
|
|
|
}
|
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_sleep(int32_t argc, Janet *argv) {
|
2019-01-06 01:45:24 +00:00
|
|
|
janet_fixarity(argc, 1);
|
2019-01-06 01:09:03 +00:00
|
|
|
double delay = janet_getnumber(argv, 0);
|
|
|
|
if (delay < 0) janet_panic("invalid argument to sleep");
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2019-02-20 01:51:34 +00:00
|
|
|
Sleep((DWORD)(delay * 1000));
|
2018-05-13 00:31:28 +00:00
|
|
|
#else
|
2018-07-07 01:50:59 +00:00
|
|
|
struct timespec ts;
|
|
|
|
ts.tv_sec = (time_t) delay;
|
|
|
|
ts.tv_nsec = (delay <= UINT32_MAX)
|
2019-02-20 01:51:34 +00:00
|
|
|
? (long)((delay - ((uint32_t)delay)) * 1000000000)
|
|
|
|
: 0;
|
2018-07-07 01:50:59 +00:00
|
|
|
nanosleep(&ts, NULL);
|
2018-05-13 00:31:28 +00:00
|
|
|
#endif
|
2019-01-06 01:09:03 +00:00
|
|
|
return janet_wrap_nil();
|
2018-03-29 00:50:20 +00:00
|
|
|
}
|
|
|
|
|
2019-01-06 01:09:03 +00:00
|
|
|
static Janet os_cwd(int32_t argc, Janet *argv) {
|
2019-01-06 01:45:24 +00:00
|
|
|
janet_fixarity(argc, 0);
|
2019-01-06 01:09:03 +00:00
|
|
|
(void) argv;
|
2018-05-19 05:09:56 +00:00
|
|
|
char buf[FILENAME_MAX];
|
|
|
|
char *ptr;
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2018-05-19 05:09:56 +00:00
|
|
|
ptr = _getcwd(buf, FILENAME_MAX);
|
|
|
|
#else
|
|
|
|
ptr = getcwd(buf, FILENAME_MAX);
|
|
|
|
#endif
|
2019-01-06 01:09:03 +00:00
|
|
|
if (NULL == ptr) janet_panic("could not get current directory");
|
|
|
|
return janet_cstringv(ptr);
|
2018-05-19 05:09:56 +00:00
|
|
|
}
|
|
|
|
|
2019-01-20 19:34:33 +00:00
|
|
|
static Janet os_date(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 0, 1);
|
|
|
|
(void) argv;
|
|
|
|
time_t t;
|
|
|
|
struct tm *t_info;
|
|
|
|
if (argc) {
|
|
|
|
t = (time_t) janet_getinteger64(argv, 0);
|
|
|
|
} else {
|
|
|
|
time(&t);
|
|
|
|
}
|
|
|
|
t_info = localtime(&t);
|
|
|
|
JanetKV *st = janet_struct_begin(9);
|
|
|
|
janet_struct_put(st, janet_ckeywordv("seconds"), janet_wrap_number(t_info->tm_sec));
|
|
|
|
janet_struct_put(st, janet_ckeywordv("minutes"), janet_wrap_number(t_info->tm_min));
|
|
|
|
janet_struct_put(st, janet_ckeywordv("hours"), janet_wrap_number(t_info->tm_hour));
|
2019-01-20 21:49:39 +00:00
|
|
|
janet_struct_put(st, janet_ckeywordv("month-day"), janet_wrap_number(t_info->tm_mday - 1));
|
2019-01-20 19:34:33 +00:00
|
|
|
janet_struct_put(st, janet_ckeywordv("month"), janet_wrap_number(t_info->tm_mon));
|
2019-01-20 21:49:39 +00:00
|
|
|
janet_struct_put(st, janet_ckeywordv("year"), janet_wrap_number(t_info->tm_year + 1900));
|
2019-01-20 19:34:33 +00:00
|
|
|
janet_struct_put(st, janet_ckeywordv("week-day"), janet_wrap_number(t_info->tm_wday));
|
|
|
|
janet_struct_put(st, janet_ckeywordv("year-day"), janet_wrap_number(t_info->tm_yday));
|
|
|
|
janet_struct_put(st, janet_ckeywordv("dst"), janet_wrap_boolean(t_info->tm_isdst));
|
|
|
|
return janet_wrap_struct(janet_struct_end(st));
|
|
|
|
}
|
|
|
|
|
2019-03-29 03:22:58 +00:00
|
|
|
static Janet os_link(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 2, 3);
|
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
(void) argc;
|
|
|
|
(void) argv;
|
|
|
|
janet_panic("os/link not supported on Windows");
|
|
|
|
return janet_wrap_nil();
|
|
|
|
#else
|
|
|
|
const char *oldpath = janet_getcstring(argv, 0);
|
|
|
|
const char *newpath = janet_getcstring(argv, 1);
|
|
|
|
int res = ((argc == 3 && janet_getboolean(argv, 2)) ? symlink : link)(oldpath, newpath);
|
2019-03-30 19:39:24 +00:00
|
|
|
if (res == -1) janet_panic(strerror(errno));
|
2019-03-29 03:22:58 +00:00
|
|
|
return janet_wrap_integer(res);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static Janet os_mkdir(int32_t argc, Janet *argv) {
|
|
|
|
janet_fixarity(argc, 1);
|
|
|
|
const char *path = janet_getcstring(argv, 0);
|
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
int res = _mkdir(path);
|
|
|
|
#else
|
|
|
|
int res = mkdir(path, S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IXGRP | S_IROTH | S_IXOTH);
|
|
|
|
#endif
|
|
|
|
return janet_wrap_boolean(res != -1);
|
|
|
|
}
|
|
|
|
|
2019-03-30 19:39:24 +00:00
|
|
|
static Janet os_rmdir(int32_t argc, Janet *argv) {
|
|
|
|
janet_fixarity(argc, 1);
|
|
|
|
const char *path = janet_getcstring(argv, 0);
|
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
int res = _rmdir(path);
|
|
|
|
#else
|
|
|
|
int res = rmdir(path);
|
|
|
|
#endif
|
|
|
|
if (res == -1) janet_panic(strerror(errno));
|
|
|
|
return janet_wrap_nil();
|
|
|
|
}
|
|
|
|
|
2019-03-29 03:22:58 +00:00
|
|
|
static Janet os_cd(int32_t argc, Janet *argv) {
|
|
|
|
janet_fixarity(argc, 1);
|
|
|
|
const char *path = janet_getcstring(argv, 0);
|
2019-03-29 03:34:24 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
int res = _chdir(path);
|
|
|
|
#else
|
2019-03-29 03:22:58 +00:00
|
|
|
int res = chdir(path);
|
2019-03-29 03:34:24 +00:00
|
|
|
#endif
|
2019-03-30 19:39:24 +00:00
|
|
|
if (res == -1) janet_panic(strerror(errno));
|
|
|
|
return janet_wrap_nil();
|
2019-03-29 03:22:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static Janet os_touch(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 1, 3);
|
|
|
|
const char *path = janet_getcstring(argv, 0);
|
|
|
|
struct utimbuf timebuf, *bufp;
|
|
|
|
if (argc >= 2) {
|
|
|
|
bufp = &timebuf;
|
|
|
|
timebuf.actime = (time_t) janet_getnumber(argv, 1);
|
|
|
|
if (argc >= 3) {
|
|
|
|
timebuf.modtime = (time_t) janet_getnumber(argv, 2);
|
|
|
|
} else {
|
|
|
|
timebuf.modtime = timebuf.actime;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
bufp = NULL;
|
|
|
|
}
|
|
|
|
int res = utime(path, bufp);
|
2019-03-30 19:39:24 +00:00
|
|
|
if (-1 == res) janet_panic(strerror(errno));
|
|
|
|
return janet_wrap_nil();
|
|
|
|
}
|
|
|
|
|
|
|
|
static Janet os_remove(int32_t argc, Janet *argv) {
|
|
|
|
janet_fixarity(argc, 1);
|
|
|
|
const char *path = janet_getcstring(argv, 0);
|
|
|
|
int status = remove(path);
|
|
|
|
if (-1 == status) janet_panic(strerror(errno));
|
|
|
|
return janet_wrap_nil();
|
2019-03-29 03:22:58 +00:00
|
|
|
}
|
|
|
|
|
2019-03-30 16:06:14 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
static const uint8_t *janet_decode_permissions(unsigned short m) {
|
|
|
|
uint8_t flags[9] = {0};
|
|
|
|
flags[0] = flags[3] = flags[6] = (m & S_IREAD) ? 'r' : '-';
|
|
|
|
flags[1] = flags[4] = flags[7] = (m & S_IWRITE) ? 'w' : '-';
|
|
|
|
flags[2] = flags[5] = flags[8] = (m & S_IEXEC) ? 'x' : '-';
|
|
|
|
return janet_string(flags, sizeof(flags));
|
|
|
|
}
|
|
|
|
|
|
|
|
static const uint8_t *janet_decode_mode(unsigned short m) {
|
|
|
|
const char *str = "other";
|
2019-03-30 17:46:52 +00:00
|
|
|
if (m & _S_IFREG) str = "file";
|
|
|
|
else if (m & _S_IFDIR) str = "directory";
|
|
|
|
else if (m & _S_IFCHR) str = "character";
|
2019-03-30 16:06:14 +00:00
|
|
|
return janet_ckeyword(str);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static const uint8_t *janet_decode_permissions(mode_t m) {
|
|
|
|
uint8_t flags[9] = {0};
|
|
|
|
flags[0] = (m & S_IRUSR) ? 'r' : '-';
|
|
|
|
flags[1] = (m & S_IWUSR) ? 'w' : '-';
|
|
|
|
flags[2] = (m & S_IXUSR) ? 'x' : '-';
|
|
|
|
flags[3] = (m & S_IRGRP) ? 'r' : '-';
|
|
|
|
flags[4] = (m & S_IWGRP) ? 'w' : '-';
|
|
|
|
flags[5] = (m & S_IXGRP) ? 'x' : '-';
|
|
|
|
flags[6] = (m & S_IROTH) ? 'r' : '-';
|
|
|
|
flags[7] = (m & S_IWOTH) ? 'w' : '-';
|
|
|
|
flags[8] = (m & S_IXOTH) ? 'x' : '-';
|
|
|
|
return janet_string(flags, sizeof(flags));
|
|
|
|
}
|
|
|
|
|
|
|
|
static const uint8_t *janet_decode_mode(mode_t m) {
|
|
|
|
const char *str = "other";
|
|
|
|
if (S_ISREG(m)) str = "file";
|
|
|
|
else if (S_ISDIR(m)) str = "directory";
|
|
|
|
else if (S_ISFIFO(m)) str = "fifo";
|
|
|
|
else if (S_ISBLK(m)) str = "block";
|
|
|
|
else if (S_ISSOCK(m)) str = "socket";
|
|
|
|
else if (S_ISLNK(m)) str = "link";
|
|
|
|
else if (S_ISCHR(m)) str = "character";
|
|
|
|
return janet_ckeyword(str);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-04-01 15:11:15 +00:00
|
|
|
/* Can we do this? */
|
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
#define stat _stat
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Getters */
|
2019-04-01 15:21:45 +00:00
|
|
|
static Janet os_stat_dev(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_dev);
|
|
|
|
}
|
|
|
|
static Janet os_stat_inode(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_ino);
|
|
|
|
}
|
|
|
|
static Janet os_stat_mode(struct stat *st) {
|
|
|
|
return janet_wrap_keyword(janet_decode_mode(st->st_mode));
|
|
|
|
}
|
|
|
|
static Janet os_stat_permissions(struct stat *st) {
|
|
|
|
return janet_wrap_string(janet_decode_permissions(st->st_mode));
|
|
|
|
}
|
|
|
|
static Janet os_stat_uid(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_uid);
|
|
|
|
}
|
|
|
|
static Janet os_stat_gid(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_gid);
|
|
|
|
}
|
|
|
|
static Janet os_stat_nlink(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_nlink);
|
|
|
|
}
|
|
|
|
static Janet os_stat_rdev(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_rdev);
|
|
|
|
}
|
|
|
|
static Janet os_stat_size(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_size);
|
|
|
|
}
|
|
|
|
static Janet os_stat_accessed(struct stat *st) {
|
|
|
|
return janet_wrap_number((double) st->st_atime);
|
|
|
|
}
|
|
|
|
static Janet os_stat_modified(struct stat *st) {
|
|
|
|
return janet_wrap_number((double) st->st_mtime);
|
|
|
|
}
|
|
|
|
static Janet os_stat_changed(struct stat *st) {
|
|
|
|
return janet_wrap_number((double) st->st_ctime);
|
|
|
|
}
|
2019-04-01 15:11:15 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2019-04-01 15:21:45 +00:00
|
|
|
static Janet os_stat_blocks(struct stat *st) {
|
|
|
|
return janet_wrap_number(0);
|
|
|
|
}
|
|
|
|
static Janet os_stat_blocksize(struct stat *st) {
|
|
|
|
return janet_wrap_number(0);
|
|
|
|
}
|
2019-04-01 15:11:15 +00:00
|
|
|
#else
|
2019-04-01 15:21:45 +00:00
|
|
|
static Janet os_stat_blocks(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_blocks);
|
|
|
|
}
|
|
|
|
static Janet os_stat_blocksize(struct stat *st) {
|
|
|
|
return janet_wrap_number(st->st_blksize);
|
|
|
|
}
|
2019-04-01 15:11:15 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
struct OsStatGetter {
|
|
|
|
const char *name;
|
2019-04-01 15:21:45 +00:00
|
|
|
Janet(*fn)(struct stat *st);
|
2019-04-01 15:11:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static const struct OsStatGetter os_stat_getters[] = {
|
|
|
|
{"dev", os_stat_dev},
|
|
|
|
{"inode", os_stat_inode},
|
|
|
|
{"mode", os_stat_mode},
|
|
|
|
{"permissions", os_stat_permissions},
|
|
|
|
{"uid", os_stat_uid},
|
|
|
|
{"gid", os_stat_gid},
|
|
|
|
{"nlink", os_stat_nlink},
|
|
|
|
{"rdev", os_stat_rdev},
|
|
|
|
{"size", os_stat_size},
|
|
|
|
{"blocks", os_stat_blocks},
|
|
|
|
{"blocksize", os_stat_blocksize},
|
|
|
|
{"accessed", os_stat_accessed},
|
|
|
|
{"modified", os_stat_modified},
|
|
|
|
{"changed", os_stat_changed},
|
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2019-03-30 16:06:14 +00:00
|
|
|
static Janet os_stat(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 1, 2);
|
|
|
|
const char *path = janet_getcstring(argv, 0);
|
2019-04-01 15:11:15 +00:00
|
|
|
JanetTable *tab = NULL;
|
|
|
|
int getall = 1;
|
|
|
|
const uint8_t *key;
|
2019-03-30 16:06:14 +00:00
|
|
|
if (argc == 2) {
|
2019-04-01 15:11:15 +00:00
|
|
|
if (janet_checktype(argv[1], JANET_KEYWORD)) {
|
|
|
|
getall = 0;
|
|
|
|
key = janet_getkeyword(argv, 1);
|
|
|
|
} else {
|
|
|
|
tab = janet_gettable(argv, 1);
|
|
|
|
}
|
2019-03-30 16:06:14 +00:00
|
|
|
} else {
|
|
|
|
tab = janet_table(0);
|
|
|
|
}
|
2019-04-01 15:11:15 +00:00
|
|
|
|
2019-03-30 16:06:14 +00:00
|
|
|
/* Build result */
|
|
|
|
struct stat st;
|
|
|
|
int res = stat(path, &st);
|
|
|
|
if (-1 == res) {
|
2019-04-05 18:45:45 +00:00
|
|
|
return janet_wrap_nil();
|
2019-03-30 16:06:14 +00:00
|
|
|
}
|
2019-04-01 15:11:15 +00:00
|
|
|
|
|
|
|
if (getall) {
|
|
|
|
/* Put results in table */
|
|
|
|
for (const struct OsStatGetter *sg = os_stat_getters; sg->name != NULL; sg++) {
|
|
|
|
janet_table_put(tab, janet_ckeywordv(sg->name), sg->fn(&st));
|
|
|
|
}
|
|
|
|
return janet_wrap_table(tab);
|
|
|
|
} else {
|
|
|
|
/* Get one result */
|
|
|
|
for (const struct OsStatGetter *sg = os_stat_getters; sg->name != NULL; sg++) {
|
|
|
|
if (janet_cstrcmp(key, sg->name)) continue;
|
|
|
|
return sg->fn(&st);
|
|
|
|
}
|
|
|
|
janet_panicf("unexpected keyword %v", janet_wrap_keyword(key));
|
|
|
|
return janet_wrap_nil();
|
|
|
|
}
|
2019-03-30 16:06:14 +00:00
|
|
|
}
|
|
|
|
|
2019-03-30 16:36:27 +00:00
|
|
|
static Janet os_dir(int32_t argc, Janet *argv) {
|
|
|
|
janet_arity(argc, 1, 2);
|
|
|
|
const char *dir = janet_getcstring(argv, 0);
|
|
|
|
JanetArray *paths = (argc == 2) ? janet_getarray(argv, 1) : janet_array(0);
|
2019-03-30 17:01:57 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
|
|
|
/* Read directory items with FindFirstFile / FindNextFile / FindClose */
|
|
|
|
struct _finddata_t afile;
|
|
|
|
char pattern[MAX_PATH + 1];
|
|
|
|
if (strlen(dir) > (sizeof(pattern) - 3))
|
|
|
|
janet_panicf("path too long: %s", dir);
|
2019-03-31 18:14:25 +00:00
|
|
|
sprintf(pattern, "%s/*", dir);
|
2019-03-30 17:01:57 +00:00
|
|
|
intptr_t res = _findfirst(pattern, &afile);
|
2019-03-30 17:46:52 +00:00
|
|
|
if (-1 == res) janet_panicv(janet_cstringv(strerror(errno)));
|
|
|
|
do {
|
|
|
|
if (strcmp(".", afile.name) && strcmp("..", afile.name)) {
|
|
|
|
janet_array_push(paths, janet_cstringv(afile.name));
|
|
|
|
}
|
|
|
|
} while (_findnext(res, &afile) != -1);
|
2019-03-30 17:01:57 +00:00
|
|
|
_findclose(res);
|
|
|
|
#else
|
|
|
|
/* Read directory items with opendir / readdir / closedir */
|
|
|
|
struct dirent *dp;
|
|
|
|
DIR *dfd = opendir(dir);
|
|
|
|
if (dfd == NULL) janet_panicf("cannot open directory %s", dir);
|
2019-03-30 16:36:27 +00:00
|
|
|
while ((dp = readdir(dfd)) != NULL) {
|
|
|
|
if (!strcmp(dp->d_name, ".") || !strcmp(dp->d_name, "..")) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
janet_array_push(paths, janet_cstringv(dp->d_name));
|
|
|
|
}
|
2019-03-30 17:01:57 +00:00
|
|
|
closedir(dfd);
|
|
|
|
#endif
|
2019-03-30 16:36:27 +00:00
|
|
|
return janet_wrap_array(paths);
|
|
|
|
}
|
|
|
|
|
2019-05-29 15:31:19 +00:00
|
|
|
static Janet os_rename(int32_t argc, Janet *argv) {
|
|
|
|
janet_fixarity(argc, 2);
|
|
|
|
const char *src = janet_getcstring(argv, 0);
|
|
|
|
const char *dest = janet_getcstring(argv, 1);
|
|
|
|
int status = rename(src, dest);
|
|
|
|
if (status) {
|
|
|
|
janet_panic(strerror(errno));
|
|
|
|
}
|
|
|
|
return janet_wrap_nil();
|
|
|
|
}
|
|
|
|
|
2019-03-29 03:22:58 +00:00
|
|
|
#endif /* JANET_REDUCED_OS */
|
|
|
|
|
2019-01-24 05:15:58 +00:00
|
|
|
static const JanetReg os_cfuns[] = {
|
2019-03-29 03:22:58 +00:00
|
|
|
{
|
|
|
|
"os/exit", os_exit,
|
|
|
|
JDOC("(os/exit x)\n\n"
|
|
|
|
"Exit from janet with an exit code equal to x. If x is not an integer, "
|
|
|
|
"the exit with status equal the hash of x.")
|
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/which", os_which,
|
|
|
|
JDOC("(os/which)\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Check the current operating system. Returns one of:\n\n"
|
|
|
|
"\t:windows - Microsoft Windows\n"
|
|
|
|
"\t:macos - Apple macos\n"
|
|
|
|
"\t:posix - A POSIX compatible system (default)")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-03-29 03:22:58 +00:00
|
|
|
{
|
|
|
|
"os/getenv", os_getenv,
|
|
|
|
JDOC("(os/getenv variable)\n\n"
|
|
|
|
"Get the string value of an environment variable.")
|
|
|
|
},
|
|
|
|
#ifndef JANET_REDUCED_OS
|
2019-03-30 16:36:27 +00:00
|
|
|
{
|
|
|
|
"os/dir", os_dir,
|
2019-05-17 08:58:06 +00:00
|
|
|
JDOC("(os/dir dir [, array])\n\n"
|
2019-03-30 16:36:27 +00:00
|
|
|
"Iterate over files and subdirectories in a directory. Returns an array of paths parts, "
|
|
|
|
"with only the filename or directory name and no prefix.")
|
|
|
|
},
|
2019-03-29 03:22:58 +00:00
|
|
|
{
|
2019-03-30 16:06:14 +00:00
|
|
|
"os/stat", os_stat,
|
2019-04-01 15:11:15 +00:00
|
|
|
JDOC("(os/stat path [, tab|key])\n\n"
|
|
|
|
"Gets information about a file or directory. Returns a table If the third argument is a keyword, returns "
|
2019-04-05 18:45:04 +00:00
|
|
|
" only that information from stat. If the file or directory does not exist, returns nil. The keys are\n\n"
|
2019-04-01 15:11:15 +00:00
|
|
|
"\t:dev - the device that the file is on\n"
|
|
|
|
"\t:mode - the type of file, one of :file, :directory, :block, :character, :fifo, :socket, :link, or :other\n"
|
|
|
|
"\t:permissions - A unix permission string like \"rwx--x--x\"\n"
|
|
|
|
"\t:uid - File uid\n"
|
|
|
|
"\t:gid - File gid\n"
|
|
|
|
"\t:nlink - number of links to file\n"
|
|
|
|
"\t:rdev - Real device of file. 0 on windows.\n"
|
|
|
|
"\t:size - size of file in bytes\n"
|
|
|
|
"\t:blocks - number of blocks in file. 0 on windows\n"
|
|
|
|
"\t:blocksize - size of blocks in file. 0 on windows\n"
|
|
|
|
"\t:accessed - timestamp when file last accessed\n"
|
|
|
|
"\t:changed - timestamp when file last chnaged (permissions changed)\n"
|
|
|
|
"\t:modified - timestamp when file last modified (content changed)\n")
|
2019-03-30 16:06:14 +00:00
|
|
|
},
|
|
|
|
{
|
2019-03-29 03:22:58 +00:00
|
|
|
"os/touch", os_touch,
|
|
|
|
JDOC("(os/touch path [, actime [, modtime]])\n\n"
|
|
|
|
"Update the access time and modification times for a file. By default, sets "
|
|
|
|
"times to the current time.")
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"os/cd", os_cd,
|
|
|
|
JDOC("(os/cd path)\n\n"
|
|
|
|
"Change current directory to path. Returns true on success, false on failure.")
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"os/mkdir", os_mkdir,
|
|
|
|
JDOC("(os/mkdir path)\n\n"
|
|
|
|
"Create a new directory. The path will be relative to the current directory if relative, otherwise "
|
|
|
|
"it will be an absolute path.")
|
|
|
|
},
|
2019-03-30 19:39:24 +00:00
|
|
|
{
|
|
|
|
"os/rmdir", os_rmdir,
|
|
|
|
JDOC("(os/rmdir path)\n\n"
|
|
|
|
"Delete a directory. The directory must be empty to succeed.")
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"os/rm", os_remove,
|
|
|
|
JDOC("(os/rm path)\n\n"
|
|
|
|
"Delete a file. Returns nil.")
|
|
|
|
},
|
2019-03-29 03:22:58 +00:00
|
|
|
{
|
|
|
|
"os/link", os_link,
|
|
|
|
JDOC("(os/link oldpath newpath [, symlink])\n\n"
|
|
|
|
"Create a symlink from oldpath to newpath. The 3 optional paramater "
|
|
|
|
"enables a hard link over a soft link. Does not work on Windows.")
|
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/execute", os_execute,
|
2019-05-30 22:40:10 +00:00
|
|
|
JDOC("(os/execute args &opts flags env)\n\n"
|
|
|
|
"Execute a program on the system and pass it string arguments. Flags "
|
|
|
|
"is a keyword that modifies how the program will execute.\n\n"
|
|
|
|
"\t:e - enables passing an environment to the program. Without :e, the "
|
|
|
|
"current environment is inherited.\n"
|
|
|
|
"\t:p - allows searching the current PATH for the binary to execute. "
|
|
|
|
"Without this flag, binaries must use absolute paths.\n\n"
|
|
|
|
"env is a table or struct mapping environment variables to values. "
|
|
|
|
"Returns the exit status of the program.")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/shell", os_shell,
|
|
|
|
JDOC("(os/shell str)\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Pass a command string str directly to the system shell.")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/setenv", os_setenv,
|
|
|
|
JDOC("(os/setenv variable value)\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Set an environment variable.")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/time", os_time,
|
|
|
|
JDOC("(os/time)\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Get the current time expressed as the number of seconds since "
|
|
|
|
"January 1, 1970, the Unix epoch. Returns a real number.")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/clock", os_clock,
|
|
|
|
JDOC("(os/clock)\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Return the number of seconds since some fixed point in time. The clock "
|
|
|
|
"is guaranteed to be non decreasing in real time.")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/sleep", os_sleep,
|
|
|
|
JDOC("(os/sleep nsec)\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Suspend the program for nsec seconds. 'nsec' can be a real number. Returns "
|
|
|
|
"nil.")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-01-06 06:49:56 +00:00
|
|
|
{
|
|
|
|
"os/cwd", os_cwd,
|
|
|
|
JDOC("(os/cwd)\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Returns the current working directory.")
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2019-01-20 19:34:33 +00:00
|
|
|
{
|
|
|
|
"os/date", os_date,
|
|
|
|
JDOC("(os/date [,time])\n\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"Returns the given time as a date struct, or the current time if no time is given. "
|
|
|
|
"Returns a struct with following key values. Note that all numbers are 0-indexed.\n\n"
|
|
|
|
"\t:seconds - number of seconds [0-61]\n"
|
|
|
|
"\t:minutes - number of minutes [0-59]\n"
|
2019-04-24 02:43:51 +00:00
|
|
|
"\t:hours - number of hours [0-23]\n"
|
2019-02-20 01:51:34 +00:00
|
|
|
"\t:month-day - day of month [0-30]\n"
|
|
|
|
"\t:month - month of year [0, 11]\n"
|
|
|
|
"\t:year - years since year 0 (e.g. 2019)\n"
|
|
|
|
"\t:week-day - day of the week [0-6]\n"
|
|
|
|
"\t:year-day - day of the year [0-365]\n"
|
|
|
|
"\t:dst - If Day Light Savings is in effect")
|
2019-01-20 19:34:33 +00:00
|
|
|
},
|
2019-05-29 15:31:19 +00:00
|
|
|
{
|
|
|
|
"os/rename", os_rename,
|
|
|
|
JDOC("(os/rename oldname newname)\n\n"
|
|
|
|
"Rename a file on disk to a new path. Returns nil.")
|
|
|
|
},
|
2019-03-29 03:22:58 +00:00
|
|
|
#endif
|
2018-11-15 20:45:41 +00:00
|
|
|
{NULL, NULL, NULL}
|
2018-03-29 00:50:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Module entry point */
|
2019-01-06 01:09:03 +00:00
|
|
|
void janet_lib_os(JanetTable *env) {
|
2019-02-08 05:44:30 +00:00
|
|
|
janet_core_cfuns(env, NULL, os_cfuns);
|
2018-03-29 00:50:20 +00:00
|
|
|
}
|