pipe
The pipe module provides pipe communication functionality, supporting both anonymous pipes and named pipes, which can be used for inter-process communication. This is an extension module of xmake.
TIP
To use this module, you need to import it first: import("core.base.pipe")
pipe.openpair
- Create an anonymous pipe pair
Function Prototype
API
pipe.openpair(mode: <string>, buffsize: <number>)Parameter Description
| Parameter | Description |
|---|---|
| mode | Optional. Pipe mode, default is "AA" |
| buffsize | Optional. Buffer size, default is 0 (system default) |
Available modes:
"BB": Both read and write in blocking mode"BA": Read blocking, write non-blocking"AB": Read non-blocking, write blocking"AA": Both read and write in non-blocking mode (default)
Usage
Creates a pair of anonymous pipes, returning read and write pipe objects.
Anonymous pipes are mainly used for communication between related processes (such as parent-child processes). A common scenario is redirecting subprocess input/output.
Basic usage example:
import("core.base.pipe")
import("core.base.bytes")
-- Create pipe pair
local rpipe, wpipe = pipe.openpair()
local buff = bytes(8192)
-- Write data
wpipe:write("hello xmake!", {block = true})
-- Read data
local read, data = rpipe:read(buff, 13)
if read > 0 and data then
print(data:str()) -- Output: hello xmake!
end
rpipe:close()
wpipe:close()Used with os.execv to redirect subprocess output:
import("core.base.pipe")
import("core.base.bytes")
-- Create pipe pair
local rpipe, wpipe = pipe.openpair()
-- Redirect subprocess stdout to pipe write end
os.execv("echo", {"hello from subprocess"}, {stdout = wpipe})
-- Close write end, read subprocess output
wpipe:close()
local buff = bytes(8192)
local read, data = rpipe:read(buff, 8192)
if read > 0 then
print("Subprocess output:", data:str())
end
rpipe:close()pipe.open
- Open a named pipe
Function Prototype
API
pipe.open(name: <string>, mode: <string>, buffsize: <number>)Parameter Description
| Parameter | Description |
|---|---|
| name | Required. Pipe name |
| mode | Required. Open mode |
| buffsize | Optional. Buffer size, default is 0 |
Available modes:
"r"or"rA": Read-only, non-blocking (client-side)"w"or"wA": Write-only, non-blocking (server-side)"rB": Read-only, blocking"wB": Write-only, blocking
Usage
Opens or creates a named pipe.
Named pipes can be used for communication between completely independent processes without any relationship. Similar to local sockets but more lightweight. Suitable for scenarios where data needs to be passed between different applications.
Server-side Example
import("core.base.pipe")
-- Open named pipe (server-side)
local pipefile = pipe.open("test", 'w')
local count = 0
while count < 10000 do
local write = pipefile:write("hello world..", {block = true})
if write <= 0 then
break
end
count = count + 1
end
print("Write successful, count:", count)
pipefile:close()Client-side Example
import("core.base.pipe")
import("core.base.bytes")
-- Open named pipe (client-side)
local pipefile = pipe.open("test", 'r')
local buff = bytes(8192)
-- Connect to server
if pipefile:connect() > 0 then
print("Connected")
local count = 0
while count < 10000 do
local read, data = pipefile:read(buff, 13, {block = true})
if read > 0 then
count = count + 1
else
break
end
end
print("Read successful, count:", count)
end
pipefile:close()pipe:read
- Read data from pipe
Function Prototype
API
pipe:read(buff: <bytes>, size: <number>, opt: <table>)Parameter Description
| Parameter | Description |
|---|---|
| buff | Required. bytes buffer object to store the read data |
| size | Required. Number of bytes to read |
| opt | Optional. Option parameters |
Options:
block: Whether to block reading, default falsestart: Buffer start position, default 1timeout: Timeout in milliseconds, default -1 (infinite wait)
Return Values
| Type | Description |
|---|---|
| read | Actual number of bytes read |
| data | Read data (bytes object) |
Usage
Reads data from the pipe into the specified buffer.
Non-blocking mode (default) returns immediately, may return 0 indicating no data available. Blocking mode waits until the specified amount of data is read or an error occurs:
import("core.base.bytes")
local buff = bytes(8192)
-- Blocking read 100 bytes, timeout 5 seconds
local read, data = rpipe:read(buff, 100, {block = true, timeout = 5000})
if read > 0 then
print("Read:", data:str())
endpipe:write
- Write data to pipe
Function Prototype
API
pipe:write(data: <string|bytes>, opt: <table>)Parameter Description
| Parameter | Description |
|---|---|
| data | Required. Data to write, can be string or bytes object |
| opt | Optional. Option parameters |
Options:
block: Whether to block writing, default falsestart: Data start position, default 1last: Data end position, default is data sizetimeout: Timeout in milliseconds, default -1
Return Value
| Type | Description |
|---|---|
| write | Actual number of bytes written |
Usage
Writes data to the pipe.
Non-blocking mode (default) may only write partial data. Blocking mode waits until all data is successfully written:
-- Blocking write data
local write = wpipe:write("hello world", {block = true})
if write > 0 then
print("Wrote", write, "bytes")
endpipe:connect
- Connect to named pipe (server-side)
Function Prototype
API
pipe:connect(opt: <table>)Parameter Description
| Parameter | Description |
|---|---|
| opt | Optional. Option parameters |
Options:
timeout: Timeout in milliseconds, default -1
Return Value
| Type | Description |
|---|---|
| number | Returns a positive number |
Usage
Connects to a named pipe, only used on the server-side of named pipes. After creating a named pipe on the server, call this method to wait for client connection.
import("core.base.pipe")
local pipefile = pipe.open("test", 'r')
if pipefile:connect() > 0 then
print("Client connected")
-- Can start reading/writing data
endpipe:wait
- Wait for pipe events
Function Prototype
API
pipe:wait(events: <number>, timeout: <number>)Parameter Description
| Parameter | Description |
|---|---|
| events | Required. Events to wait for |
| timeout | Required. Timeout in milliseconds, -1 means wait indefinitely |
Event constants:
pipe.EV_READ(1): Readable event, indicates pipe has data to readpipe.EV_WRITE(2): Writable event, indicates pipe can accept datapipe.EV_CONN(2): Connection event, used for named pipes to wait for client connection
Return Value
| Type | Description |
|---|---|
| number | Returns the actual event constant value that occurred |
Usage
Waits for specified pipe events to occur. In non-blocking mode, this method can be used to implement event-driven I/O.
In non-blocking mode, wait can be used to implement efficient event loops:
-- Wait for pipe to be readable, timeout 1 second
local events = rpipe:wait(pipe.EV_READ, 1000)
if events == pipe.EV_READ then
-- Pipe is readable, can read data
local read, data = rpipe:read(buff, 100)
end
-- Wait for pipe to be writable
local events = wpipe:wait(pipe.EV_WRITE, 1000)
if events == pipe.EV_WRITE then
-- Pipe is writable, can write data
wpipe:write("data")
endpipe:close
- Close the pipe
Function Prototype
API
pipe:close()Parameter Description
No parameters required for this function.
Usage
Closes the pipe and releases resources. Pipes should be closed promptly after use.
pipe:name
- Get pipe name
Function Prototype
API
pipe:name()Parameter Description
No parameters required for this function.
Usage
Gets the name of a named pipe. Returns nil for anonymous pipes.
TIP
Pipes are unidirectional, one end can only read, the other can only write. For bidirectional communication, two pipes are needed.
WARNING
Remember to call close() after using the pipe to release resources. A bytes buffer must be created using bytes() before reading data.