mirror of
https://github.com/vlang/v.git
synced 2023-08-10 21:13:21 +03:00
net: make net.socket.write, net.socket.read_line more robust
This commit is contained in:
parent
1ccd1979a4
commit
c73f34cc5f
@ -9,3 +9,7 @@ module net
|
|||||||
fn error_code() int {
|
fn error_code() int {
|
||||||
return C.errno
|
return C.errno
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub const (
|
||||||
|
MSG_NOSIGNAL = 0x4000
|
||||||
|
)
|
||||||
|
@ -32,3 +32,6 @@ fn error_code() int {
|
|||||||
return C.WSAGetLastError()
|
return C.WSAGetLastError()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub const (
|
||||||
|
MSG_NOSIGNAL = 0
|
||||||
|
)
|
||||||
|
@ -188,7 +188,7 @@ pub fn dial(address string, port int) ?Socket {
|
|||||||
|
|
||||||
// send string data to socket
|
// send string data to socket
|
||||||
pub fn (s Socket) send(buf byteptr, len int) ?int {
|
pub fn (s Socket) send(buf byteptr, len int) ?int {
|
||||||
res := int( C.send(s.sockfd, buf, len, 0) )
|
res := int( C.send(s.sockfd, buf, len, MSG_NOSIGNAL) )
|
||||||
if res < 0 {
|
if res < 0 {
|
||||||
return error('net.send: failed with $res')
|
return error('net.send: failed with $res')
|
||||||
}
|
}
|
||||||
@ -241,52 +241,59 @@ pub fn (s Socket) close() ?int {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
pub const (
|
||||||
MAX_READ = 400
|
CRLF = '\r\n'
|
||||||
|
MAX_READ = 400
|
||||||
|
MSG_PEEK = 0x02
|
||||||
)
|
)
|
||||||
pub fn (s Socket) write(str string) {
|
|
||||||
line := '$str\r\n'
|
// write - write a string with CRLF after it over the socket s
|
||||||
C.send(s.sockfd, line.str, line.len, 0)
|
pub fn (s Socket) write(str string) ?int {
|
||||||
|
line := '$str$CRLF'
|
||||||
|
res := int( C.send(s.sockfd, line.str, line.len, MSG_NOSIGNAL) )
|
||||||
|
if res < 0 { return error('net.write: failed with $res') }
|
||||||
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// read_line - retrieves a line from the socket s (i.e. a string ended with \n)
|
||||||
pub fn (s Socket) read_line() string {
|
pub fn (s Socket) read_line() string {
|
||||||
mut res := ''
|
mut buf := [MAX_READ]byte // where C.recv will store the network data
|
||||||
for {
|
mut res := '' // The final result, including the ending \n.
|
||||||
$if debug {
|
for {
|
||||||
println('.')
|
mut line := '' // The current line. Can be a partial without \n in it.
|
||||||
}
|
n := int(C.recv(s.sockfd, buf, MAX_READ-1, MSG_PEEK))
|
||||||
mut buf := malloc(MAX_READ)
|
if n == -1 { return res }
|
||||||
n := int(C.recv(s.sockfd, buf, MAX_READ-1, 0))
|
if n == 0 { return res }
|
||||||
$if debug {
|
buf[n] = `\0`
|
||||||
println('numbytes=$n')
|
mut eol_idx := -1
|
||||||
|
for i := 0; i < n; i++ {
|
||||||
|
if int(buf[i]) == `\n` {
|
||||||
|
eol_idx = i
|
||||||
|
// Ensure that tos_clone(buf) later,
|
||||||
|
// will return *only* the first line (including \n),
|
||||||
|
// and ignore the rest
|
||||||
|
buf[i+1] = `\0`
|
||||||
|
break
|
||||||
}
|
}
|
||||||
if n == -1 {
|
}
|
||||||
$if debug {
|
line = tos_clone(buf)
|
||||||
println('recv failed')
|
if eol_idx > 0 {
|
||||||
}
|
// At this point, we are sure that recv returned valid data,
|
||||||
// TODO
|
// that contains *at least* one line.
|
||||||
return ''
|
// Ensure that the block till the first \n (including it)
|
||||||
}
|
// is removed from the socket's receive queue, so that it does
|
||||||
if n == 0 {
|
// not get read again.
|
||||||
break
|
C.recv(s.sockfd, buf, eol_idx+1, 0)
|
||||||
}
|
res += line
|
||||||
// println('resp len=$numbytes')
|
break
|
||||||
buf[n] = `\0`
|
}
|
||||||
// C.printf('!!buf= "%s" n=%d\n', buf,n)
|
// recv returned a buffer without \n in it .
|
||||||
line := string(buf)
|
C.recv(s.sockfd, buf, n, 0)
|
||||||
res += line
|
res += line
|
||||||
// Reached a newline. That's an end of an IRC message
|
res += CRLF
|
||||||
// TODO dont need ends_with check ?
|
break
|
||||||
if line.ends_with('\n') || n < MAX_READ - 1 {
|
}
|
||||||
// println('NL')
|
return res
|
||||||
break
|
|
||||||
}
|
|
||||||
if line.ends_with('\r\n') {
|
|
||||||
// println('RNL')
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return res
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (s Socket) get_port() int {
|
pub fn (s Socket) get_port() int {
|
||||||
|
@ -1,21 +1,23 @@
|
|||||||
import net
|
import net
|
||||||
|
|
||||||
fn test_socket() {
|
fn setup() (net.Socket, net.Socket, net.Socket) {
|
||||||
server := net.listen(0) or {
|
server := net.listen(0) or { panic(err) }
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
server_port := server.get_port()
|
server_port := server.get_port()
|
||||||
client := net.dial('127.0.0.1', server_port) or {
|
client := net.dial('127.0.0.1', server_port) or { panic(err) }
|
||||||
panic(err)
|
socket := server.accept() or { panic(err) }
|
||||||
}
|
return server, client, socket
|
||||||
socket := server.accept() or {
|
}
|
||||||
panic(err)
|
|
||||||
}
|
fn cleanup(server &net.Socket, client &net.Socket, socket &net.Socket) {
|
||||||
|
server.close() or {}
|
||||||
|
client.close() or {}
|
||||||
|
socket.close() or {}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn test_socket() {
|
||||||
|
server, client, socket := setup()
|
||||||
message := 'Hello World'
|
message := 'Hello World'
|
||||||
socket.send(message.str, message.len) or {
|
socket.send(message.str, message.len) or { assert false }
|
||||||
assert false
|
|
||||||
}
|
|
||||||
$if debug { println('message send: $message') }
|
$if debug { println('message send: $message') }
|
||||||
$if debug { println('send socket: $socket.sockfd') }
|
$if debug { println('send socket: $socket.sockfd') }
|
||||||
|
|
||||||
@ -23,10 +25,33 @@ fn test_socket() {
|
|||||||
received := tos(bytes, blen)
|
received := tos(bytes, blen)
|
||||||
$if debug { println('message received: $received') }
|
$if debug { println('message received: $received') }
|
||||||
$if debug { println('client: $client.sockfd') }
|
$if debug { println('client: $client.sockfd') }
|
||||||
|
|
||||||
assert message == received
|
assert message == received
|
||||||
|
cleanup(server, client, socket)
|
||||||
server.close() or {}
|
}
|
||||||
client.close() or {}
|
|
||||||
socket.close() or {}
|
fn test_socket_write() {
|
||||||
|
server, client, socket := setup()
|
||||||
|
message1 := 'a message 1'
|
||||||
|
socket.write(message1) or { assert false }
|
||||||
|
line1 := client.read_line()
|
||||||
|
assert line1 != message1
|
||||||
|
assert line1.trim_space() == message1
|
||||||
|
cleanup(server, client, socket)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn test_socket_write_fail_without_panic() {
|
||||||
|
server, client, socket := setup()
|
||||||
|
message2 := 'a message 2'
|
||||||
|
// ensure that socket.write (i.e. done on the server side)
|
||||||
|
// continues to work, even when the client side has been disconnected
|
||||||
|
// this test is important for a stable long standing server
|
||||||
|
client.close() or {}
|
||||||
|
for i:=0; i<3; i++{
|
||||||
|
socket.write(message2) or {
|
||||||
|
println('write to a socket without a recipient should produce an option fail: $err | $message2')
|
||||||
|
assert true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cleanup(server, client, socket)
|
||||||
}
|
}
|
||||||
|
@ -42,5 +42,7 @@ pub fn (b mut Builder) str() string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn (b mut Builder) free() {
|
pub fn (b mut Builder) free() {
|
||||||
//free(b.buf.data)
|
unsafe{ free(b.buf.data) }
|
||||||
|
b.buf = make(0, 1, 1)
|
||||||
|
b.len = 0
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user