mirror of
https://github.com/gechandesu/runcmd.git
synced 2026-02-13 08:41:51 +03:00
Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 8286af90eb | |||
| 14864ba992 | |||
| c62722cba9 | |||
| 7acab2165e | |||
| 8e7d757b3b |
@@ -9,7 +9,7 @@ but I don't like any of them. So let's overview.
|
|||||||
|
|
||||||
* `os.execvp()`, `os.execve()` — cross-platform versions of the C functions of the same name.
|
* `os.execvp()`, `os.execve()` — cross-platform versions of the C functions of the same name.
|
||||||
|
|
||||||
* `os.execute()`, `os.execute_opt()`, `os.execute_or_exit()`, `os.execute_or_panic()` — starts and waits for a command to completed. Under the hood, they perform a dirty hack by calling shell with stream redirection `'exec 2>&1;${cmd}'`.
|
* `os.execute()`, `os.execute_opt()`, `os.execute_or_exit()`, `os.execute_or_panic()`, `os.raw_execute()` — starts and waits for a command to completed. Under the hood, they perform a dirty hack by calling shell with stream redirection `'exec 2>&1;${cmd}'`.
|
||||||
|
|
||||||
* `util.execute_with_timeout()` (from `os.util`) — just an `os.execute()` wrapper.
|
* `util.execute_with_timeout()` (from `os.util`) — just an `os.execute()` wrapper.
|
||||||
|
|
||||||
@@ -55,7 +55,6 @@ directory. **Examples are very important**.
|
|||||||
|
|
||||||
- [x] Basic implementation.
|
- [x] Basic implementation.
|
||||||
- [x] Contexts support for creating cancelable commands, commands with timeouts, etc.
|
- [x] Contexts support for creating cancelable commands, commands with timeouts, etc.
|
||||||
- [ ] Process groups support, pgkill().
|
|
||||||
- [ ] Better error handling and more tests...
|
- [ ] Better error handling and more tests...
|
||||||
|
|
||||||
Send pull requests for additional features/bugfixes.
|
Send pull requests for additional features/bugfixes.
|
||||||
|
|||||||
26
bytebuf.v
26
bytebuf.v
@@ -1,26 +0,0 @@
|
|||||||
module runcmd
|
|
||||||
|
|
||||||
import io
|
|
||||||
|
|
||||||
// buffer creates simple bytes buffer that can be read through `io.Reader` interface.
|
|
||||||
pub fn buffer(data []u8) ByteBuffer {
|
|
||||||
return ByteBuffer{
|
|
||||||
bytes: data
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ByteBuffer {
|
|
||||||
bytes []u8
|
|
||||||
mut:
|
|
||||||
pos int
|
|
||||||
}
|
|
||||||
|
|
||||||
// read reads `buf.len` bytes from internal bytes buffer and returns number of bytes read.
|
|
||||||
pub fn (mut b ByteBuffer) read(mut buf []u8) !int {
|
|
||||||
if b.pos >= b.bytes.len {
|
|
||||||
return io.Eof{}
|
|
||||||
}
|
|
||||||
n := copy(mut buf, b.bytes[b.pos..])
|
|
||||||
b.pos += n
|
|
||||||
return n
|
|
||||||
}
|
|
||||||
2
cmd.v
2
cmd.v
@@ -268,7 +268,7 @@ pub fn (mut c Command) start() !int {
|
|||||||
path: path
|
path: path
|
||||||
argv: c.args
|
argv: c.args
|
||||||
env: if c.env.len == 0 { os.environ() } else { c.env }
|
env: if c.env.len == 0 { os.environ() } else { c.env }
|
||||||
dir: c.dir
|
dir: os.abs_path(c.dir)
|
||||||
post_fork: [parent_pipes_hook]
|
post_fork: [parent_pipes_hook]
|
||||||
pre_exec: pre_exec_hooks
|
pre_exec: pre_exec_hooks
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import io.string_reader
|
import io
|
||||||
import rand
|
import rand
|
||||||
import runcmd
|
import runcmd
|
||||||
import time
|
import time
|
||||||
@@ -20,7 +20,7 @@ fn main() {
|
|||||||
mut child_stdout := cmd.stdout()!
|
mut child_stdout := cmd.stdout()!
|
||||||
|
|
||||||
// Prepare reader to store command output.
|
// Prepare reader to store command output.
|
||||||
mut output := string_reader.StringReader.new(reader: child_stdout)
|
mut output := io.new_buffered_reader(reader: child_stdout)
|
||||||
|
|
||||||
// Start stdout reading in a coroutine.
|
// Start stdout reading in a coroutine.
|
||||||
//
|
//
|
||||||
@@ -56,4 +56,6 @@ fn main() {
|
|||||||
|
|
||||||
// wait() will close the child stdout file desciptor by itself.
|
// wait() will close the child stdout file desciptor by itself.
|
||||||
cmd.wait()!
|
cmd.wait()!
|
||||||
|
|
||||||
|
println('Child state: ${cmd.state}')
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import io.string_reader
|
import io
|
||||||
import runcmd
|
import runcmd
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
@@ -14,7 +14,7 @@ fn main() {
|
|||||||
|
|
||||||
// Setup StringReader with stdout input. Note the cmd.stdout()! call, it
|
// Setup StringReader with stdout input. Note the cmd.stdout()! call, it
|
||||||
// returns the io.Reader interface and reads child process stdout file descriptor.
|
// returns the io.Reader interface and reads child process stdout file descriptor.
|
||||||
mut reader := string_reader.StringReader.new(reader: cmd.stdout()!)
|
mut reader := io.new_buffered_reader(reader: cmd.stdout()!)
|
||||||
|
|
||||||
// Read sdtout line by line until EOF.
|
// Read sdtout line by line until EOF.
|
||||||
for {
|
for {
|
||||||
|
|||||||
@@ -1,6 +1,23 @@
|
|||||||
|
import io
|
||||||
import strings
|
import strings
|
||||||
import runcmd
|
import runcmd
|
||||||
|
|
||||||
|
struct ByteBuffer {
|
||||||
|
bytes []u8
|
||||||
|
mut:
|
||||||
|
pos int
|
||||||
|
}
|
||||||
|
|
||||||
|
// read reads `buf.len` bytes from internal bytes buffer and returns number of bytes read.
|
||||||
|
pub fn (mut b ByteBuffer) read(mut buf []u8) !int {
|
||||||
|
if b.pos >= b.bytes.len {
|
||||||
|
return io.Eof{}
|
||||||
|
}
|
||||||
|
n := copy(mut buf, b.bytes[b.pos..])
|
||||||
|
b.pos += n
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
input := 'Hello from parent process!'
|
input := 'Hello from parent process!'
|
||||||
|
|
||||||
@@ -8,8 +25,10 @@ fn main() {
|
|||||||
// * `reader` reads input from the parent process; it will be copied to the
|
// * `reader` reads input from the parent process; it will be copied to the
|
||||||
// standard input of the child process.
|
// standard input of the child process.
|
||||||
// * `writer` accepts data from the child process; it will be copied from the
|
// * `writer` accepts data from the child process; it will be copied from the
|
||||||
// standard output of the child process.
|
// standard output of the child process. This is optinal.
|
||||||
mut reader := runcmd.buffer(input.bytes())
|
mut reader := ByteBuffer{
|
||||||
|
bytes: input.bytes()
|
||||||
|
}
|
||||||
mut writer := strings.new_builder(4096)
|
mut writer := strings.new_builder(4096)
|
||||||
|
|
||||||
// Prepare the command.
|
// Prepare the command.
|
||||||
|
|||||||
2
v.mod
2
v.mod
@@ -1,7 +1,7 @@
|
|||||||
Module {
|
Module {
|
||||||
name: 'runcmd'
|
name: 'runcmd'
|
||||||
description: 'Run external commands'
|
description: 'Run external commands'
|
||||||
version: '0.3.0'
|
version: '0.4.0'
|
||||||
license: 'Unlicense'
|
license: 'Unlicense'
|
||||||
repo_url: 'https://github.com/gechandesu/runcmd'
|
repo_url: 'https://github.com/gechandesu/runcmd'
|
||||||
dependencies: []
|
dependencies: []
|
||||||
|
|||||||
Reference in New Issue
Block a user