178 lines
4.8 KiB
Go
Raw Normal View History

2024-01-03 20:09:47 +09:00
package sockets
2023-06-26 03:16:43 -05:00
import (
"context"
"io"
"net"
"os"
"syscall/js"
"time"
"github.com/syumai/workers/internal/jsutil"
)
2024-01-03 21:20:07 +09:00
func newSocket(ctx context.Context, sockVal js.Value, readDeadline, writeDeadline time.Time) *Socket {
2024-01-03 20:55:26 +09:00
ctx, cancel := context.WithCancel(ctx)
2024-01-03 21:20:07 +09:00
writerVal := sockVal.Get("writable").Call("getWriter")
readerVal := sockVal.Get("readable")
readCloser := jsutil.ConvertReadableStreamToReadCloser(readerVal)
2024-01-03 21:20:07 +09:00
return &Socket{
2024-01-03 20:55:26 +09:00
ctx: ctx,
cancel: cancel,
2024-01-03 21:20:07 +09:00
reader: readCloser,
2024-01-03 21:20:07 +09:00
writerVal: writerVal,
readDeadline: readDeadline,
writeDeadline: writeDeadline,
startTLS: func() js.Value { return sockVal.Call("startTls") },
close: func() { sockVal.Call("close") },
closeRead: func() { readCloser.Close() },
2024-01-03 21:20:07 +09:00
closeWrite: func() { writerVal.Call("close") },
2024-01-03 20:55:26 +09:00
}
2023-06-26 03:16:43 -05:00
}
2024-01-03 20:40:52 +09:00
type Socket struct {
2024-01-03 21:20:07 +09:00
ctx context.Context
cancel context.CancelFunc
2023-06-26 03:16:43 -05:00
2024-01-03 21:20:07 +09:00
reader io.Reader
writerVal js.Value
2023-06-26 03:16:43 -05:00
readDeadline time.Time
writeDeadline time.Time
2024-01-03 21:20:07 +09:00
startTLS func() js.Value
close func()
closeRead func()
closeWrite func()
2023-06-26 03:16:43 -05:00
}
2024-01-03 20:40:52 +09:00
var _ net.Conn = (*Socket)(nil)
2024-01-03 18:36:46 +09:00
2023-06-26 03:16:43 -05:00
// Read reads data from the connection.
// Read can be made to time out and return an error after a fixed
// time limit; see SetDeadline and SetReadDeadline.
2024-01-03 20:40:52 +09:00
func (t *Socket) Read(b []byte) (n int, err error) {
2024-01-03 18:37:21 +09:00
ctx, cancel := context.WithDeadline(t.ctx, t.readDeadline)
defer cancel()
2023-06-26 03:22:29 -05:00
done := make(chan struct{})
go func() {
2024-01-03 21:20:07 +09:00
n, err = t.reader.Read(b)
2023-06-26 03:22:29 -05:00
close(done)
}()
select {
case <-done:
return
case <-ctx.Done():
return 0, os.ErrDeadlineExceeded
}
2023-06-26 03:16:43 -05:00
}
// Write writes data to the connection.
// Write can be made to time out and return an error after a fixed
// time limit; see SetDeadline and SetWriteDeadline.
2024-01-03 20:40:52 +09:00
func (t *Socket) Write(b []byte) (n int, err error) {
2024-01-03 18:37:21 +09:00
ctx, cancel := context.WithDeadline(t.ctx, t.writeDeadline)
defer cancel()
2023-06-26 03:16:43 -05:00
done := make(chan struct{})
go func() {
arr := jsutil.NewUint8Array(len(b))
js.CopyBytesToJS(arr, b)
2024-01-03 21:20:07 +09:00
_, err = jsutil.AwaitPromise(t.writerVal.Call("write", arr))
// TODO: handle error
2023-06-26 03:16:43 -05:00
if err == nil {
n = len(b)
}
close(done)
}()
select {
case <-done:
return
case <-ctx.Done():
return 0, os.ErrDeadlineExceeded
}
}
2024-01-03 21:20:07 +09:00
// StartTLS upgrades an insecure socket to a secure one that uses TLS, returning a new *Socket.
func (t *Socket) StartTLS() *Socket {
return newSocket(t.ctx, t.startTLS(), t.readDeadline, t.writeDeadline)
2023-06-26 03:27:30 -05:00
}
2023-06-26 03:16:43 -05:00
// Close closes the connection.
// Any blocked Read or Write operations will be unblocked and return errors.
2024-01-03 20:40:52 +09:00
func (t *Socket) Close() error {
2024-01-03 21:20:07 +09:00
defer t.cancel()
t.close()
2023-06-26 03:16:43 -05:00
return nil
}
// CloseRead closes the read side of the connection.
2024-01-03 20:40:52 +09:00
func (t *Socket) CloseRead() error {
2024-01-03 21:20:07 +09:00
t.closeRead()
2023-06-26 03:16:43 -05:00
return nil
}
// CloseWrite closes the write side of the connection.
2024-01-03 20:40:52 +09:00
func (t *Socket) CloseWrite() error {
2024-01-03 21:20:07 +09:00
t.closeWrite()
2023-06-26 03:16:43 -05:00
return nil
}
// LocalAddr returns the local network address, if known.
2024-01-03 20:40:52 +09:00
func (t *Socket) LocalAddr() net.Addr {
2023-06-26 03:16:43 -05:00
return nil
}
// RemoteAddr returns the remote network address, if known.
2024-01-03 20:40:52 +09:00
func (t *Socket) RemoteAddr() net.Addr {
2023-06-26 03:16:43 -05:00
return nil
}
// SetDeadline sets the read and write deadlines associated
// with the connection. It is equivalent to calling both
// SetReadDeadline and SetWriteDeadline.
//
// A deadline is an absolute time after which I/O operations
// fail instead of blocking. The deadline applies to all future
// and pending I/O, not just the immediately following call to
// Read or Write. After a deadline has been exceeded, the
// connection can be refreshed by setting a deadline in the future.
//
// If the deadline is exceeded a call to Read or Write or to other
// I/O methods will return an error that wraps os.ErrDeadlineExceeded.
// This can be tested using errors.Is(err, os.ErrDeadlineExceeded).
// The error's Timeout method will return true, but note that there
// are other possible errors for which the Timeout method will
// return true even if the deadline has not been exceeded.
//
// An idle timeout can be implemented by repeatedly extending
// the deadline after successful Read or Write calls.
//
// A zero value for t means I/O operations will not time out.
2024-01-03 20:40:52 +09:00
func (t *Socket) SetDeadline(deadline time.Time) error {
2023-06-26 03:16:43 -05:00
t.SetReadDeadline(deadline)
t.SetWriteDeadline(deadline)
return nil
}
// SetReadDeadline sets the deadline for future Read calls
// and any currently-blocked Read call.
// A zero value for t means Read will not time out.
2024-01-03 20:40:52 +09:00
func (t *Socket) SetReadDeadline(deadline time.Time) error {
2023-06-26 03:22:29 -05:00
t.readDeadline = deadline
2023-06-26 03:16:43 -05:00
return nil
}
// SetWriteDeadline sets the deadline for future Write calls
// and any currently-blocked Write call.
// Even if write times out, it may return n > 0, indicating that
// some of the data was successfully written.
// A zero value for t means Write will not time out.
2024-01-03 20:40:52 +09:00
func (t *Socket) SetWriteDeadline(deadline time.Time) error {
2023-06-26 03:16:43 -05:00
t.writeDeadline = deadline
return nil
}