open-nomad/api/fs.go

149 lines
3.6 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
)
// 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-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
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
}
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
}
}