308 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			V
		
	
	
			
		
		
	
	
			308 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			V
		
	
	
| module os
 | |
| 
 | |
| import strings
 | |
| 
 | |
| #include <dirent.h>
 | |
| #include <unistd.h>
 | |
| #include <fcntl.h>
 | |
| #include <sys/utsname.h>
 | |
| 
 | |
| pub const (
 | |
| 	path_separator = '/'
 | |
| 	path_delimiter = ':'
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	stdin_value  = 0
 | |
| 	stdout_value = 1
 | |
| 	stderr_value = 2
 | |
| )
 | |
| 
 | |
| // (Must be realized in Syscall) (Must be specified)
 | |
| // ref: http://www.ccfit.nsu.ru/~deviv/courses/unix/unix/ng7c229.html
 | |
| pub const (
 | |
| 	s_ifmt  = 0xF000 // type of file
 | |
| 	s_ifdir = 0x4000 // directory
 | |
| 	s_iflnk = 0xa000 // link
 | |
| 	s_isuid = 0o4000 // SUID
 | |
| 	s_isgid = 0o2000 // SGID
 | |
| 	s_isvtx = 0o1000 // Sticky
 | |
| 	s_irusr = 0o0400 // Read by owner
 | |
| 	s_iwusr = 0o0200 // Write by owner
 | |
| 	s_ixusr = 0o0100 // Execute by owner
 | |
| 	s_irgrp = 0o0040 // Read by group
 | |
| 	s_iwgrp = 0o0020 // Write by group
 | |
| 	s_ixgrp = 0o0010 // Execute by group
 | |
| 	s_iroth = 0o0004 // Read by others
 | |
| 	s_iwoth = 0o0002 // Write by others
 | |
| 	s_ixoth = 0o0001 // Execute by others
 | |
| )
 | |
| 
 | |
| struct C.utsname {
 | |
| mut:
 | |
| 	sysname  charptr
 | |
| 	nodename charptr
 | |
| 	release  charptr
 | |
| 	version  charptr
 | |
| 	machine  charptr
 | |
| }
 | |
| 
 | |
| fn C.uname(name voidptr) int
 | |
| 
 | |
| fn C.symlink(charptr, charptr) int
 | |
| 
 | |
| pub fn uname() Uname {
 | |
| 	mut u := Uname{}
 | |
| 	utsize := sizeof(C.utsname)
 | |
| 	x := malloc(int(utsize))
 | |
| 	d := &C.utsname(x)
 | |
| 	if C.uname(d) == 0 {
 | |
| 		u.sysname = cstring_to_vstring(byteptr(d.sysname))
 | |
| 		u.nodename = cstring_to_vstring(byteptr(d.nodename))
 | |
| 		u.release = cstring_to_vstring(byteptr(d.release))
 | |
| 		u.version = cstring_to_vstring(byteptr(d.version))
 | |
| 		u.machine = cstring_to_vstring(byteptr(d.machine))
 | |
| 	}
 | |
| 	free(d)
 | |
| 	return u
 | |
| }
 | |
| 
 | |
| fn init_os_args(argc int, argv &&byte) []string {
 | |
| 	mut args_ := []string{}
 | |
| 	// mut args := []string(make(0, argc, sizeof(string)))
 | |
| 	// mut args := []string{len:argc}
 | |
| 	for i in 0 .. argc {
 | |
| 		// args [i] = argv[i].vstring()
 | |
| 		unsafe { args_ << byteptr(argv[i]).vstring() }
 | |
| 	}
 | |
| 	return args_
 | |
| }
 | |
| 
 | |
| pub fn ls(path string) ?[]string {
 | |
| 	mut res := []string{}
 | |
| 	dir := C.opendir(charptr(path.str))
 | |
| 	if isnil(dir) {
 | |
| 		return error('ls() couldnt open dir "$path"')
 | |
| 	}
 | |
| 	mut ent := &C.dirent(0)
 | |
| 	// mut ent := &C.dirent{!}
 | |
| 	for {
 | |
| 		ent = C.readdir(dir)
 | |
| 		if isnil(ent) {
 | |
| 			break
 | |
| 		}
 | |
| 		bptr := byteptr(ent.d_name)
 | |
| 		unsafe {
 | |
| 			if bptr[0] == 0 || (bptr[0] == `.` && bptr[1] == 0)
 | |
| 				|| (bptr[0] == `.` && bptr[1] == `.` && bptr[2] == 0) {
 | |
| 				continue
 | |
| 			}
 | |
| 		}
 | |
| 		res << tos_clone(bptr)
 | |
| 	}
 | |
| 	C.closedir(dir)
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| /*
 | |
| pub fn is_dir(path string) bool {
 | |
| 	//$if linux {
 | |
| 		//C.syscall(4, path.str) // sys_newstat
 | |
| 	//}
 | |
| 	dir := C.opendir(path.str)
 | |
| 	res := !isnil(dir)
 | |
| 	if res {
 | |
| 		C.closedir(dir)
 | |
| 	}
 | |
| 	return res
 | |
| }
 | |
| */
 | |
| /*
 | |
| pub fn (mut f File) fseek(pos, mode int) {
 | |
| }
 | |
| */
 | |
| // mkdir creates a new directory with the specified path.
 | |
| pub fn mkdir(path string) ?bool {
 | |
| 	if path == '.' {
 | |
| 		return true
 | |
| 	}
 | |
| 	/*
 | |
| 	mut k := 0
 | |
| 	defer {
 | |
| 		k = 1
 | |
| 	}
 | |
| 	*/
 | |
| 	apath := real_path(path)
 | |
| 	// defer {
 | |
| 	// apath.free()
 | |
| 	//}
 | |
| 	/*
 | |
| 	$if linux {
 | |
| 		$if !android {
 | |
| 			ret := C.syscall(sys_mkdir, apath.str, 511)
 | |
| 			if ret == -1 {
 | |
| 				return error(posix_get_error_msg(C.errno))
 | |
| 			}
 | |
| 			return true
 | |
| 		}
 | |
| 	}
 | |
| 	*/
 | |
| 	r := unsafe { C.mkdir(charptr(apath.str), 511) }
 | |
| 	if r == -1 {
 | |
| 		return error(posix_get_error_msg(C.errno))
 | |
| 	}
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| // exec starts the specified command, waits for it to complete, and returns its output.
 | |
| pub fn exec(cmd string) ?Result {
 | |
| 	// if cmd.contains(';') || cmd.contains('&&') || cmd.contains('||') || cmd.contains('\n') {
 | |
| 	// return error(';, &&, || and \\n are not allowed in shell commands')
 | |
| 	// }
 | |
| 	pcmd := '$cmd 2>&1'
 | |
| 	f := vpopen(pcmd)
 | |
| 	if isnil(f) {
 | |
| 		return error('exec("$cmd") failed')
 | |
| 	}
 | |
| 	buf := [4096]byte{}
 | |
| 	mut res := strings.new_builder(1024)
 | |
| 	for C.fgets(charptr(buf), 4096, f) != 0 {
 | |
| 		bufbp := byteptr(buf)
 | |
| 		res.write_bytes(bufbp, vstrlen(bufbp))
 | |
| 	}
 | |
| 	soutput := res.str()
 | |
| 	// res.free()
 | |
| 	exit_code := vpclose(f)
 | |
| 	if exit_code == 127 {
 | |
| 		return error_with_code(soutput, 127)
 | |
| 	}
 | |
| 	return Result{
 | |
| 		exit_code: exit_code
 | |
| 		output: soutput
 | |
| 	}
 | |
| }
 | |
| 
 | |
| pub struct Command {
 | |
| mut:
 | |
| 	f voidptr
 | |
| pub mut:
 | |
| 	eof bool
 | |
| pub:
 | |
| 	path            string
 | |
| 	redirect_stdout bool
 | |
| }
 | |
| 
 | |
| // pub fn command(cmd Command) Command {
 | |
| //}
 | |
| pub fn (mut c Command) start() ? {
 | |
| 	pcmd := '$c.path 2>&1'
 | |
| 	c.f = vpopen(pcmd)
 | |
| 	if isnil(c.f) {
 | |
| 		return error('exec("$c.path") failed')
 | |
| 	}
 | |
| }
 | |
| 
 | |
| pub fn (mut c Command) read_line() string {
 | |
| 	buf := [4096]byte{}
 | |
| 	mut res := strings.new_builder(1024)
 | |
| 	unsafe {
 | |
| 		for C.fgets(charptr(buf), 4096, c.f) != 0 {
 | |
| 			bufbp := byteptr(buf)
 | |
| 			len := vstrlen(bufbp)
 | |
| 			for i in 0 .. len {
 | |
| 				if int(bufbp[i]) == `\n` {
 | |
| 					res.write_bytes(bufbp, i)
 | |
| 					return res.str()
 | |
| 				}
 | |
| 			}
 | |
| 			res.write_bytes(bufbp, len)
 | |
| 		}
 | |
| 	}
 | |
| 	c.eof = true
 | |
| 	return res.str()
 | |
| }
 | |
| 
 | |
| pub fn (c &Command) close() ? {
 | |
| 	exit_code := vpclose(c.f)
 | |
| 	if exit_code == 127 {
 | |
| 		return error_with_code('error', 127)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| pub fn symlink(origin string, target string) ?bool {
 | |
| 	res := C.symlink(charptr(origin.str), charptr(target.str))
 | |
| 	if res == 0 {
 | |
| 		return true
 | |
| 	}
 | |
| 	return error(posix_get_error_msg(C.errno))
 | |
| }
 | |
| 
 | |
| // get_error_msg return error code representation in string.
 | |
| pub fn get_error_msg(code int) string {
 | |
| 	return posix_get_error_msg(code)
 | |
| }
 | |
| 
 | |
| pub fn (mut f File) close() {
 | |
| 	if !f.is_opened {
 | |
| 		return
 | |
| 	}
 | |
| 	f.is_opened = false
 | |
| 	/*
 | |
| 	$if linux {
 | |
| 		$if !android {
 | |
| 			C.syscall(sys_close, f.fd)
 | |
| 			return
 | |
| 		}
 | |
| 	}
 | |
| 	*/
 | |
| 	C.fflush(f.cfile)
 | |
| 	C.fclose(f.cfile)
 | |
| }
 | |
| 
 | |
| pub fn debugger_present() bool {
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| fn C.mkstemp(stemplate byteptr) int
 | |
| 
 | |
| // `is_writable_folder` - `folder` exists and is writable to the process
 | |
| pub fn is_writable_folder(folder string) ?bool {
 | |
| 	if !exists(folder) {
 | |
| 		return error('`$folder` does not exist')
 | |
| 	}
 | |
| 	if !is_dir(folder) {
 | |
| 		return error('`folder` is not a folder')
 | |
| 	}
 | |
| 	tmp_perm_check := join_path(folder, 'XXXXXX')
 | |
| 	unsafe {
 | |
| 		x := C.mkstemp(charptr(tmp_perm_check.str))
 | |
| 		if -1 == x {
 | |
| 			return error('folder `$folder` is not writable')
 | |
| 		}
 | |
| 		C.close(x)
 | |
| 	}
 | |
| 	rm(tmp_perm_check) ?
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| [inline]
 | |
| pub fn getpid() int {
 | |
| 	return C.getpid()
 | |
| }
 | |
| 
 | |
| // Turns the given bit on or off, depending on the `enable` parameter
 | |
| pub fn posix_set_permission_bit(path_s string, mode u32, enable bool) {
 | |
| 	mut s := C.stat{}
 | |
| 	mut new_mode := u32(0)
 | |
| 	path := charptr(path_s.str)
 | |
| 	unsafe {
 | |
| 		C.stat(path, &s)
 | |
| 		new_mode = s.st_mode
 | |
| 	}
 | |
| 	match enable {
 | |
| 		true { new_mode |= mode }
 | |
| 		false { new_mode &= (0o7777 - mode) }
 | |
| 	}
 | |
| 	C.chmod(path, int(new_mode))
 | |
| }
 |