open-nomad/api/fs.go

439 lines
10 KiB
Go
Raw Normal View History

2016-01-26 22:31:52 +00:00
package api
import (
"encoding/json"
"fmt"
2016-01-27 00:07:59 +00:00
"io"
"io/ioutil"
2016-01-26 22:31:52 +00:00
"net/http"
"net/url"
2016-01-27 00:07:59 +00:00
"strconv"
2016-01-27 22:20:10 +00:00
"time"
2016-01-26 22:31:52 +00:00
)
2016-07-07 18:51:40 +00:00
const (
// OriginStart and OriginEnd are the available parameters for the origin
// argument when streaming a file. They respectively offset from the start
// and end of a file.
OriginStart = "start"
OriginEnd = "end"
)
// AllocFileInfo holds information about a file inside the AllocDir
type AllocFileInfo struct {
2016-01-27 22:20:10 +00:00
Name string
IsDir bool
Size int64
FileMode string
ModTime time.Time
}
2016-07-07 18:51:40 +00:00
// StreamFrame is used to frame data of a file when streaming
type StreamFrame struct {
2016-07-10 22:56:13 +00:00
Offset int64 `json:",omitempty"`
Data []byte `json:",omitempty"`
File string `json:",omitempty"`
FileEvent string `json:",omitempty"`
2016-07-07 18:51:40 +00:00
}
2016-07-10 22:56:13 +00:00
// IsHeartbeat returns if the frame is a heartbeat frame
2016-07-07 18:51:40 +00:00
func (s *StreamFrame) IsHeartbeat() bool {
2016-07-10 22:56:13 +00:00
return len(s.Data) == 0 && s.FileEvent == "" && s.File == "" && s.Offset == 0
2016-07-07 18:51:40 +00:00
}
2016-01-26 22:44:33 +00:00
// AllocFS is used to introspect an allocation directory on a Nomad client
2016-01-26 22:31:52 +00:00
type AllocFS struct {
client *Client
}
2016-01-26 22:44:33 +00:00
// AllocFS returns an handle to the AllocFS endpoints
2016-01-26 22:31:52 +00:00
func (c *Client) AllocFS() *AllocFS {
return &AllocFS{client: c}
}
2016-01-26 22:44:33 +00:00
// List is used to list the files at a given path of an allocation directory
func (a *AllocFS) List(alloc *Allocation, path string, q *QueryOptions) ([]*AllocFileInfo, *QueryMeta, error) {
2016-01-26 22:31:52 +00:00
node, _, err := a.client.Nodes().Info(alloc.NodeID, &QueryOptions{})
if err != nil {
return nil, nil, err
}
if node.HTTPAddr == "" {
return nil, nil, fmt.Errorf("http addr of the node where alloc %q is running is not advertised", alloc.ID)
}
u := &url.URL{
Scheme: "http",
Host: node.HTTPAddr,
2016-01-26 22:44:33 +00:00
Path: fmt.Sprintf("/v1/client/fs/ls/%s", alloc.ID),
2016-01-26 22:31:52 +00:00
}
v := url.Values{}
v.Set("path", path)
u.RawQuery = v.Encode()
req := &http.Request{
Method: "GET",
URL: u,
}
c := http.Client{}
resp, err := c.Do(req)
if err != nil {
return nil, nil, err
}
if resp.StatusCode != 200 {
return nil, nil, a.getErrorMsg(resp)
}
2016-01-26 22:31:52 +00:00
decoder := json.NewDecoder(resp.Body)
var files []*AllocFileInfo
2016-01-26 22:31:52 +00:00
if err := decoder.Decode(&files); err != nil {
return nil, nil, err
}
return files, nil, nil
}
2016-01-26 23:03:26 +00:00
// Stat is used to stat a file at a given path of an allocation directory
func (a *AllocFS) Stat(alloc *Allocation, path string, q *QueryOptions) (*AllocFileInfo, *QueryMeta, error) {
2016-01-26 23:03:26 +00:00
node, _, err := a.client.Nodes().Info(alloc.NodeID, &QueryOptions{})
if err != nil {
return nil, nil, err
}
if node.HTTPAddr == "" {
return nil, nil, fmt.Errorf("http addr of the node where alloc %q is running is not advertised", alloc.ID)
}
u := &url.URL{
Scheme: "http",
Host: node.HTTPAddr,
Path: fmt.Sprintf("/v1/client/fs/stat/%s", alloc.ID),
}
v := url.Values{}
v.Set("path", path)
u.RawQuery = v.Encode()
req := &http.Request{
Method: "GET",
URL: u,
}
c := http.Client{}
resp, err := c.Do(req)
if err != nil {
return nil, nil, err
}
if resp.StatusCode != 200 {
return nil, nil, a.getErrorMsg(resp)
}
2016-01-26 23:03:26 +00:00
decoder := json.NewDecoder(resp.Body)
var file *AllocFileInfo
2016-01-26 23:03:26 +00:00
if err := decoder.Decode(&file); err != nil {
return nil, nil, err
}
return file, nil, nil
}
2016-01-27 00:07:59 +00:00
// ReadAt is used to read bytes at a given offset until limit at the given path
// in an allocation directory. If limit is <= 0, there is no limit.
2016-07-13 19:23:33 +00:00
func (a *AllocFS) ReadAt(alloc *Allocation, path string, offset int64, limit int64, q *QueryOptions) (io.ReadCloser, *QueryMeta, error) {
2016-01-27 00:07:59 +00:00
node, _, err := a.client.Nodes().Info(alloc.NodeID, &QueryOptions{})
if err != nil {
2016-01-27 20:30:27 +00:00
return nil, nil, err
2016-01-27 00:07:59 +00:00
}
if node.HTTPAddr == "" {
2016-01-27 20:30:27 +00:00
return nil, nil, fmt.Errorf("http addr of the node where alloc %q is running is not advertised", alloc.ID)
2016-01-27 00:07:59 +00:00
}
u := &url.URL{
Scheme: "http",
Host: node.HTTPAddr,
Path: fmt.Sprintf("/v1/client/fs/readat/%s", alloc.ID),
}
v := url.Values{}
v.Set("path", path)
v.Set("offset", strconv.FormatInt(offset, 10))
v.Set("limit", strconv.FormatInt(limit, 10))
u.RawQuery = v.Encode()
req := &http.Request{
Method: "GET",
URL: u,
}
c := http.Client{}
resp, err := c.Do(req)
if err != nil {
2016-01-27 20:30:27 +00:00
return nil, nil, err
2016-01-27 00:07:59 +00:00
}
2016-01-27 20:30:27 +00:00
return resp.Body, nil, nil
2016-01-27 00:07:59 +00:00
}
2016-03-28 18:06:22 +00:00
// Cat is used to read contents of a file at the given path in an allocation
// directory
func (a *AllocFS) Cat(alloc *Allocation, path string, q *QueryOptions) (io.ReadCloser, *QueryMeta, error) {
2016-03-28 18:06:22 +00:00
node, _, err := a.client.Nodes().Info(alloc.NodeID, &QueryOptions{})
if err != nil {
return nil, nil, err
}
if node.HTTPAddr == "" {
return nil, nil, fmt.Errorf("http addr of the node where alloc %q is running is not advertised", alloc.ID)
}
u := &url.URL{
Scheme: "http",
Host: node.HTTPAddr,
Path: fmt.Sprintf("/v1/client/fs/cat/%s", alloc.ID),
}
v := url.Values{}
v.Set("path", path)
u.RawQuery = v.Encode()
req := &http.Request{
Method: "GET",
URL: u,
}
c := http.Client{}
resp, err := c.Do(req)
if err != nil {
return nil, nil, err
}
return resp.Body, nil, nil
}
func (a *AllocFS) getErrorMsg(resp *http.Response) error {
if errMsg, err := ioutil.ReadAll(resp.Body); err == nil {
return fmt.Errorf(string(errMsg))
} else {
return err
}
}
2016-07-07 18:51:40 +00:00
2016-07-10 22:56:13 +00:00
// Stream streams the content of a file blocking on EOF.
// The parameters are:
// * path: path to file to stream.
// * offset: The offset to start streaming data at.
// * origin: Either "start" or "end" and defines from where the offset is applied.
2016-07-18 18:39:38 +00:00
// * cancel: A channel that when closed, streaming will end.
2016-07-10 22:56:13 +00:00
//
// The return value is a channel that will emit StreamFrames as they are read.
2016-07-07 18:51:40 +00:00
func (a *AllocFS) Stream(alloc *Allocation, path, origin string, offset int64,
cancel <-chan struct{}, q *QueryOptions) (<-chan *StreamFrame, *QueryMeta, error) {
node, _, err := a.client.Nodes().Info(alloc.NodeID, q)
if err != nil {
return nil, nil, err
}
if node.HTTPAddr == "" {
return nil, nil, fmt.Errorf("http addr of the node where alloc %q is running is not advertised", alloc.ID)
}
u := &url.URL{
Scheme: "http",
Host: node.HTTPAddr,
Path: fmt.Sprintf("/v1/client/fs/stream/%s", alloc.ID),
}
v := url.Values{}
v.Set("path", path)
v.Set("origin", origin)
v.Set("offset", strconv.FormatInt(offset, 10))
u.RawQuery = v.Encode()
req := &http.Request{
Method: "GET",
URL: u,
Cancel: cancel,
}
c := http.Client{}
resp, err := c.Do(req)
if err != nil {
return nil, nil, err
}
// Create the output channel
frames := make(chan *StreamFrame, 10)
2016-07-18 18:39:38 +00:00
go func() {
// Close the body
defer resp.Body.Close()
// Create a decoder
dec := json.NewDecoder(resp.Body)
for {
// Check if we have been cancelled
select {
case <-cancel:
return
default:
}
// Decode the next frame
var frame StreamFrame
if err := dec.Decode(&frame); err != nil {
close(frames)
return
}
// Discard heartbeat frames
if frame.IsHeartbeat() {
continue
}
frames <- &frame
}
}()
return frames, nil, nil
}
// Logs streams the content of a tasks logs blocking on EOF.
// The parameters are:
// * allocation: the allocation to stream from.
// * task: the tasks name to stream logs for.
// * logType: Either "stdout" or "stderr"
// * offset: The offset to start streaming data at.
// * origin: Either "start" or "end" and defines from where the offset is applied.
// * cancel: A channel that when closed, streaming will end.
//
// The return value is a channel that will emit StreamFrames as they are read.
func (a *AllocFS) Logs(alloc *Allocation, task, logType, origin string, offset int64,
cancel <-chan struct{}, q *QueryOptions) (<-chan *StreamFrame, *QueryMeta, error) {
node, _, err := a.client.Nodes().Info(alloc.NodeID, q)
if err != nil {
return nil, nil, err
}
if node.HTTPAddr == "" {
return nil, nil, fmt.Errorf("http addr of the node where alloc %q is running is not advertised", alloc.ID)
}
u := &url.URL{
Scheme: "http",
Host: node.HTTPAddr,
Path: fmt.Sprintf("/v1/client/fs/logs/%s", alloc.ID),
}
v := url.Values{}
v.Set("task", task)
v.Set("type", logType)
v.Set("origin", origin)
v.Set("offset", strconv.FormatInt(offset, 10))
u.RawQuery = v.Encode()
req := &http.Request{
Method: "GET",
URL: u,
Cancel: cancel,
}
c := http.Client{}
resp, err := c.Do(req)
if err != nil {
return nil, nil, err
}
// Create the output channel
frames := make(chan *StreamFrame, 10)
2016-07-07 18:51:40 +00:00
go func() {
// Close the body
defer resp.Body.Close()
// Create a decoder
dec := json.NewDecoder(resp.Body)
for {
// Check if we have been cancelled
select {
case <-cancel:
return
default:
}
// Decode the next frame
var frame StreamFrame
if err := dec.Decode(&frame); err != nil {
close(frames)
return
}
// Discard heartbeat frames
if frame.IsHeartbeat() {
continue
}
frames <- &frame
}
}()
return frames, nil, nil
}
2016-07-12 23:29:18 +00:00
// FrameReader is used to convert a stream of frames into a read closer.
type FrameReader struct {
frames <-chan *StreamFrame
cancelCh chan struct{}
closed bool
2016-07-12 23:29:18 +00:00
unblockTime time.Duration
2016-07-12 23:29:18 +00:00
frame *StreamFrame
frameOffset int
byteOffset int
}
// NewFrameReader takes a channel of frames and returns a FrameReader which
// implements io.ReadCloser
func NewFrameReader(frames <-chan *StreamFrame, cancelCh chan struct{}) *FrameReader {
return &FrameReader{
frames: frames,
cancelCh: cancelCh,
}
}
// SetUnblockTime sets the time to unblock and return zero bytes read. If the
// duration is unset or is zero or less, the read will block til data is read.
func (f *FrameReader) SetUnblockTime(d time.Duration) {
f.unblockTime = d
}
2016-07-12 23:29:18 +00:00
// Offset returns the offset into the stream.
func (f *FrameReader) Offset() int {
return f.byteOffset
}
// Read reads the data of the incoming frames into the bytes buffer. Returns EOF
// when there are no more frames.
func (f *FrameReader) Read(p []byte) (n int, err error) {
if f.frame == nil {
var unblock <-chan time.Time
if f.unblockTime.Nanoseconds() > 0 {
unblock = time.After(f.unblockTime)
2016-07-12 23:29:18 +00:00
}
select {
case frame, ok := <-f.frames:
if !ok {
return 0, io.EOF
}
f.frame = frame
// Store the total offset into the file
f.byteOffset = int(f.frame.Offset)
case <-unblock:
return 0, nil
}
2016-07-12 23:29:18 +00:00
}
// Copy the data out of the frame and update our offset
n = copy(p, f.frame.Data[f.frameOffset:])
f.frameOffset += n
// Clear the frame and its offset once we have read everything
if len(f.frame.Data) == f.frameOffset {
f.frame = nil
f.frameOffset = 0
}
return n, nil
}
// Close cancels the stream of frames
func (f *FrameReader) Close() error {
if f.closed {
return nil
}
2016-07-12 23:29:18 +00:00
close(f.cancelCh)
f.closed = true
2016-07-12 23:29:18 +00:00
return nil
}