Add command and shell :by_line()

This commit is contained in:
Bruce Hill 2025-03-19 16:17:44 -04:00
parent e7dc0ec85c
commit f3b8529e01
3 changed files with 111 additions and 1 deletions

View File

@ -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

View File

@ -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)!)

View File

@ -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