Add command and shell :by_line()
This commit is contained in:
parent
e7dc0ec85c
commit
f3b8529e01
@ -11,10 +11,11 @@
|
|||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/wait.h>
|
#include <sys/wait.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <unistr.h>
|
||||||
|
|
||||||
#define READ_END 0
|
#define READ_END 0
|
||||||
#define WRITE_END 1
|
#define WRITE_END 1
|
||||||
public int run_command(Text_t exe, Array_t arg_array, Table_t env_table,
|
int run_command(Text_t exe, Array_t arg_array, Table_t env_table,
|
||||||
Array_t input_bytes, Array_t *output_bytes, Array_t *error_bytes)
|
Array_t input_bytes, Array_t *output_bytes, Array_t *error_bytes)
|
||||||
{
|
{
|
||||||
pthread_testcancel();
|
pthread_testcancel();
|
||||||
@ -173,5 +174,103 @@ public int run_command(Text_t exe, Array_t arg_array, Table_t env_table,
|
|||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
pid_t pid;
|
||||||
|
FILE *out;
|
||||||
|
} child_info_t;
|
||||||
|
|
||||||
|
static void _line_reader_cleanup(child_info_t *child)
|
||||||
|
{
|
||||||
|
if (child && child->out) {
|
||||||
|
fclose(child->out);
|
||||||
|
child->out = NULL;
|
||||||
|
}
|
||||||
|
if (child->pid) {
|
||||||
|
kill(child->pid, SIGTERM);
|
||||||
|
child->pid = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static Text_t _next_line(child_info_t *child)
|
||||||
|
{
|
||||||
|
if (!child || !child->out) return NONE_TEXT;
|
||||||
|
|
||||||
|
char *line = NULL;
|
||||||
|
size_t size = 0;
|
||||||
|
ssize_t len = getline(&line, &size, child->out);
|
||||||
|
if (len <= 0) {
|
||||||
|
_line_reader_cleanup(child);
|
||||||
|
return NONE_TEXT;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (len > 0 && (line[len-1] == '\r' || line[len-1] == '\n'))
|
||||||
|
--len;
|
||||||
|
|
||||||
|
if (u8_check((uint8_t*)line, (size_t)len) != NULL)
|
||||||
|
fail("Invalid UTF8!");
|
||||||
|
|
||||||
|
Text_t line_text = Text$from_strn(line, len);
|
||||||
|
free(line);
|
||||||
|
return line_text;
|
||||||
|
}
|
||||||
|
|
||||||
|
OptionalClosure_t command_by_line(Text_t exe, Array_t arg_array, Table_t env_table)
|
||||||
|
{
|
||||||
|
posix_spawnattr_t attr;
|
||||||
|
posix_spawnattr_init(&attr);
|
||||||
|
|
||||||
|
int child_outpipe[2];
|
||||||
|
pipe(child_outpipe);
|
||||||
|
|
||||||
|
posix_spawn_file_actions_t actions;
|
||||||
|
posix_spawn_file_actions_init(&actions);
|
||||||
|
posix_spawn_file_actions_adddup2(&actions, child_outpipe[WRITE_END], STDOUT_FILENO);
|
||||||
|
posix_spawn_file_actions_addclose(&actions, child_outpipe[READ_END]);
|
||||||
|
|
||||||
|
const char *exe_str = Text$as_c_string(exe);
|
||||||
|
|
||||||
|
Array_t arg_strs = {};
|
||||||
|
Array$insert_value(&arg_strs, exe_str, I(0), sizeof(char*));
|
||||||
|
for (int64_t i = 0; i < arg_array.length; i++)
|
||||||
|
Array$insert_value(&arg_strs, Text$as_c_string(*(Text_t*)(arg_array.data + i*arg_array.stride)), I(0), sizeof(char*));
|
||||||
|
Array$insert_value(&arg_strs, NULL, I(0), sizeof(char*));
|
||||||
|
char **args = arg_strs.data;
|
||||||
|
|
||||||
|
extern char **environ;
|
||||||
|
char **env = environ;
|
||||||
|
if (env_table.entries.length > 0) {
|
||||||
|
Array_t env_array = {}; // Array of const char*
|
||||||
|
for (char **e = environ; *e; e++)
|
||||||
|
Array$insert(&env_array, e, I(0), sizeof(char*));
|
||||||
|
|
||||||
|
for (int64_t i = 0; i < env_table.entries.length; i++) {
|
||||||
|
struct { Text_t key, value; } *entry = env_table.entries.data + env_table.entries.stride*i;
|
||||||
|
const char *env_entry = heap_strf("%k=%k", &entry->key, &entry->value);
|
||||||
|
Array$insert(&env_array, &env_entry, I(0), sizeof(char*));
|
||||||
|
}
|
||||||
|
Array$insert_value(&env_array, NULL, I(0), sizeof(char*));
|
||||||
|
assert(env_array.stride == sizeof(char*));
|
||||||
|
env = env_array.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
pid_t pid;
|
||||||
|
int ret = exe_str[0] == '/' ?
|
||||||
|
posix_spawn(&pid, exe_str, &actions, &attr, args, env)
|
||||||
|
: posix_spawnp(&pid, exe_str, &actions, &attr, args, env);
|
||||||
|
if (ret != 0)
|
||||||
|
return NONE_CLOSURE;
|
||||||
|
|
||||||
|
posix_spawnattr_destroy(&attr);
|
||||||
|
posix_spawn_file_actions_destroy(&actions);
|
||||||
|
|
||||||
|
close(child_outpipe[WRITE_END]);
|
||||||
|
|
||||||
|
child_info_t *child_info = GC_MALLOC(sizeof(child_info_t));
|
||||||
|
child_info->out = fdopen(child_outpipe[READ_END], "r");
|
||||||
|
child_info->pid = pid;
|
||||||
|
GC_register_finalizer(child_info, (void*)_line_reader_cleanup, NULL, NULL, NULL);
|
||||||
|
return (Closure_t){.fn=(void*)_next_line, .userdata=child_info};
|
||||||
|
}
|
||||||
|
|
||||||
#undef READ_END
|
#undef READ_END
|
||||||
#undef WRITE_END
|
#undef WRITE_END
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
# Functions for running system commands
|
# Functions for running system commands
|
||||||
|
|
||||||
use ./commands.c
|
use ./commands.c
|
||||||
|
use libunistring.so
|
||||||
|
|
||||||
extern run_command:func(exe:Text, args:[Text], env:{Text,Text}, input:[Byte], output:&[Byte], error:&[Byte] -> Int32)
|
extern run_command:func(exe:Text, args:[Text], env:{Text,Text}, input:[Byte], output:&[Byte], error:&[Byte] -> Int32)
|
||||||
|
extern command_by_line:func(exe:Text, args:[Text], env:{Text,Text} -> func(->Text?)?)
|
||||||
|
|
||||||
enum ExitType(Exited(status:Int32), Signaled(signal:Int32), Failed)
|
enum ExitType(Exited(status:Int32), Signaled(signal:Int32), Failed)
|
||||||
|
|
||||||
@ -61,6 +63,9 @@ struct Command(command:Text, args=[:Text], env={:Text,Text}):
|
|||||||
return none
|
return none
|
||||||
else: return none
|
else: return none
|
||||||
|
|
||||||
|
func by_line(command:Command -> func(->Text?)?):
|
||||||
|
return command_by_line(command.command, command.args, command.env)
|
||||||
|
|
||||||
func main(command:Text, args:[Text], input=""):
|
func main(command:Text, args:[Text], input=""):
|
||||||
cmd := Command(command, args)
|
cmd := Command(command, args)
|
||||||
say(cmd:get_output(input=input)!)
|
say(cmd:get_output(input=input)!)
|
||||||
|
@ -33,3 +33,9 @@ lang Shell:
|
|||||||
func get_output_bytes(shell:Shell, input="", input_bytes=[:Byte] -> [Byte]?):
|
func get_output_bytes(shell:Shell, input="", input_bytes=[:Byte] -> [Byte]?):
|
||||||
return shell:command():get_output_bytes(input=input, input_bytes=input_bytes)
|
return shell:command():get_output_bytes(input=input, input_bytes=input_bytes)
|
||||||
|
|
||||||
|
func by_line(shell:Shell -> func(->Text?)?):
|
||||||
|
return shell:command():by_line()
|
||||||
|
|
||||||
|
func main(command:Shell):
|
||||||
|
for line in command:by_line()!:
|
||||||
|
>> line
|
||||||
|
Loading…
Reference in New Issue
Block a user