v/vlib/os/os_nix.v

219 lines
3.9 KiB
V
Raw Normal View History

module os
2019-07-01 17:04:09 +02:00
#include <dirent.h>
#include <unistd.h>
2019-10-24 11:36:57 +02:00
pub const (
2019-10-12 21:18:19 +02:00
path_separator = '/'
)
2019-12-31 08:53:53 +01:00
const (
stdin_value = 0
stdout_value = 1
stderr_value = 2
)
2019-12-27 19:10:06 +01:00
fn C.symlink(charptr, charptr) int
2019-10-31 11:08:01 +01:00
pub fn init_os_args(argc int, argv &byteptr) []string {
2019-09-14 22:48:30 +02:00
mut args := []string
2019-10-13 00:50:15 +02:00
for i in 0 .. argc {
2019-09-14 22:48:30 +02:00
args << string(argv[i])
2019-11-29 17:14:26 +01:00
}
2019-09-14 22:48:30 +02:00
return args
}
// get_error_msg return error code representation in string.
pub fn get_error_msg(code int) string {
2019-11-24 04:27:02 +01:00
ptr_text := C.strerror(code) // voidptr?
if ptr_text == 0 {
return ''
}
2019-11-24 04:27:02 +01:00
return tos3(ptr_text)
2019-07-29 18:21:36 +02:00
}
2019-08-16 14:05:11 +02:00
2019-10-17 13:30:05 +02:00
pub fn ls(path string) ?[]string {
2019-08-16 14:05:11 +02:00
mut res := []string
dir := C.opendir(path.str)
2019-08-16 14:05:11 +02:00
if isnil(dir) {
2019-10-17 13:30:05 +02:00
return error('ls() couldnt open dir "$path"')
2019-08-16 14:05:11 +02:00
}
2019-12-04 10:19:32 +01:00
mut ent := &C.dirent(0)
2019-12-19 22:29:37 +01:00
// mut ent := &C.dirent{!}
2019-08-16 14:05:11 +02:00
for {
ent = C.readdir(dir)
if isnil(ent) {
break
}
2019-09-15 18:07:40 +02:00
name := tos_clone(byteptr(ent.d_name))
2019-08-16 14:05:11 +02:00
if name != '.' && name != '..' && name != '' {
res << name
}
}
C.closedir(dir)
return res
}
2019-08-17 15:17:43 +02:00
/*
pub fn is_dir(path string) bool {
//$if linux {
//C.syscall(4, path.str) // sys_newstat
//}
2019-08-17 15:17:43 +02:00
dir := C.opendir(path.str)
res := !isnil(dir)
if res {
C.closedir(dir)
}
return res
}
*/
2019-08-17 15:17:43 +02:00
2019-12-31 08:53:53 +01:00
pub fn open(path string) ?File {
$if linux {
2019-12-31 19:53:15 +01:00
//$if linux_or_macos {
2019-12-31 08:53:53 +01:00
fd := C.syscall(sys_open, path.str, 511)
if fd == -1 {
return error('failed to open file "$path"')
}
return File{
fd: fd
opened: true
}
}
$else {
cpath := path.str
file := File{
cfile: C.fopen(charptr(cpath), 'rb')
opened: true
}
if isnil(file.cfile) {
return error('failed to open file "$path"')
}
return file
}
}
// create creates a file at a specified location and returns a writable `File` object.
pub fn create(path string) ?File {
$if linux {
2019-12-31 19:53:15 +01:00
//$if linux_or_macos {
mut fd := 0
2020-01-01 10:15:05 +01:00
//println('creat SYS')
2019-12-31 19:53:15 +01:00
$if macos {
fd = C.syscall(sys_open_nocancel, path.str, 0x601, 0x1b6)
}
$else {
fd = C.syscall(sys_creat, path.str, 511)
}
2020-01-01 10:15:05 +01:00
//println('fd=$fd')
2019-12-31 08:53:53 +01:00
if fd == -1 {
return error('failed to create file "$path"')
}
return File{
fd: fd
opened: true
}
}
mut file := File{
cfile: C.fopen(charptr(path.str), 'wb')
opened: true
}
if isnil(file.cfile) {
return error('failed to create file "$path"')
}
return file
}
/*
pub fn (f mut File) fseek(pos, mode int) {
}
*/
2019-12-31 08:53:53 +01:00
pub fn (f mut File) write(s string) {
if !f.opened {
return
}
$if linux {
2019-12-31 19:53:15 +01:00
//$if linux_or_macos {
2019-12-31 08:53:53 +01:00
C.syscall(sys_write, f.fd, s.str, s.len)
return
}
C.fputs(s.str, f.cfile)
// C.fwrite(s.str, 1, s.len, f.cfile)
}
pub fn (f mut File) writeln(s string) {
if !f.opened {
return
}
2019-12-31 19:53:15 +01:00
//$if linux_or_macos {
2019-12-31 08:53:53 +01:00
$if linux {
snl := s + '\n'
C.syscall(sys_write, f.fd, snl.str, snl.len)
return
}
// C.fwrite(s.str, 1, s.len, f.cfile)
// ss := s.clone()
// TODO perf
C.fputs(s.str, f.cfile)
// ss.free()
C.fputs('\n', f.cfile)
}
2019-08-17 15:17:43 +02:00
// mkdir creates a new directory with the specified path.
2019-11-23 17:55:18 +01:00
pub fn mkdir(path string) ?bool {
2019-12-19 22:29:37 +01:00
if path == '.' {
return true
}
apath := os.realpath(path)
2019-12-31 08:53:53 +01:00
$if linux {
ret := C.syscall(sys_mkdir, apath.str, 511)
if ret == -1 {
return error(get_error_msg(C.errno))
}
return true
}
2019-12-07 13:51:00 +01:00
r := C.mkdir(apath.str, 511)
2019-11-23 17:55:18 +01:00
if r == -1 {
return error(get_error_msg(C.errno))
}
return true
2019-08-17 15:17:43 +02:00
}
2019-11-07 14:01:17 +01:00
// exec starts the specified command, waits for it to complete, and returns its output.
pub fn exec(cmd string) ?Result {
2019-12-19 22:29:37 +01:00
// if cmd.contains(';') || cmd.contains('&&') || cmd.contains('||') || cmd.contains('\n') {
// return error(';, &&, || and \\n are not allowed in shell commands')
// }
2019-11-07 14:01:17 +01:00
pcmd := '$cmd 2>&1'
f := vpopen(pcmd)
if isnil(f) {
return error('exec("$cmd") failed')
}
buf := [1000]byte
mut res := ''
2019-12-01 08:33:26 +01:00
for C.fgets(charptr(buf), 1000, f) != 0 {
2019-11-07 14:01:17 +01:00
res += tos(buf, vstrlen(buf))
}
res = res.trim_space()
exit_code := vpclose(f)
2019-12-19 22:29:37 +01:00
// if exit_code != 0 {
// return error(res)
// }
return Result{
2019-11-07 14:01:17 +01:00
output: res
exit_code: exit_code
}
}
2019-12-19 22:29:37 +01:00
2019-12-27 19:10:06 +01:00
pub fn symlink(origin, target string) ?bool {
res := C.symlink(origin.str, target.str)
if res == 0 { return true }
return error(get_error_msg(C.errno))
}