2016-01-26 22:31:52 +00:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2016-01-27 00:07:59 +00:00
|
|
|
"io"
|
2016-01-28 05:39:50 +00:00
|
|
|
"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"
|
|
|
|
)
|
|
|
|
|
2016-01-27 19:22:51 +00:00
|
|
|
// 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-01-27 19:22:51 +00:00
|
|
|
}
|
|
|
|
|
2016-07-07 18:51:40 +00:00
|
|
|
// StreamFrame is used to frame data of a file when streaming
|
|
|
|
type StreamFrame struct {
|
|
|
|
Offset int64
|
|
|
|
Data string
|
|
|
|
File string
|
|
|
|
FileEvent string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *StreamFrame) IsHeartbeat() bool {
|
2016-07-09 16:28:46 +00:00
|
|
|
return s.Data == "" && 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
|
2016-01-27 19:22:51 +00:00
|
|
|
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
|
|
|
|
}
|
2016-01-28 05:39:50 +00:00
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, nil, a.getErrorMsg(resp)
|
|
|
|
}
|
2016-01-26 22:31:52 +00:00
|
|
|
decoder := json.NewDecoder(resp.Body)
|
2016-01-27 19:22:51 +00:00
|
|
|
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
|
2016-01-27 19:22:51 +00:00
|
|
|
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
|
|
|
|
}
|
2016-01-28 05:39:50 +00:00
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, nil, a.getErrorMsg(resp)
|
|
|
|
}
|
2016-01-26 23:03:26 +00:00
|
|
|
decoder := json.NewDecoder(resp.Body)
|
2016-01-27 19:22:51 +00:00
|
|
|
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
|
2016-07-09 16:28:46 +00:00
|
|
|
// in an allocation directory. If limit is <= 0, there is no limit.
|
2016-01-27 20:30:27 +00:00
|
|
|
func (a *AllocFS) ReadAt(alloc *Allocation, path string, offset int64, limit int64, q *QueryOptions) (io.Reader, *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-01-28 05:39:50 +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.Reader, *QueryMeta, error) {
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2016-01-28 05:39:50 +00:00
|
|
|
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
|
|
|
|
|
|
|
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)
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|