4 Commits
v2 ... password

Author SHA1 Message Date
Ovv
20d2802b76 Implement passwords app list 2019-12-21 16:12:26 +01:00
Adphi
76b69b7acd Fix user in doc.go [skip-ci] 2019-07-30 18:52:32 +00:00
90081d6e8f improved comments 2019-07-23 09:52:34 +02:00
Adphi
5d284c1ad8 Merge branch 'v2' into 'master'
V2

See merge request partitio/Nextcloud-Partitio/gonextcloud!4
2019-07-22 14:56:11 +00:00
25 changed files with 196 additions and 73 deletions

2
app.go
View File

@@ -1,6 +1,6 @@
package gonextcloud package gonextcloud
//App // App is a nextcloud application (plugin)
type App struct { type App struct {
ID string `json:"id"` ID string `json:"id"`
Ocsid string `json:"ocsid"` Ocsid string `json:"ocsid"`

View File

@@ -13,7 +13,7 @@ type apps struct {
//List return the list of the Nextcloud apps //List return the list of the Nextcloud apps
func (a *apps) List() ([]string, error) { func (a *apps) List() ([]string, error) {
res, err := a.c.baseRequest(http.MethodGet, routes.apps, nil) res, err := a.c.baseOcsRequest(http.MethodGet, routes.apps, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -27,7 +27,7 @@ func (a *apps) ListEnabled() ([]string, error) {
ro := &req.RequestOptions{ ro := &req.RequestOptions{
Params: map[string]string{"filter": "enabled"}, Params: map[string]string{"filter": "enabled"},
} }
res, err := a.c.baseRequest(http.MethodGet, routes.apps, ro) res, err := a.c.baseOcsRequest(http.MethodGet, routes.apps, ro)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -41,7 +41,7 @@ func (a *apps) ListDisabled() ([]string, error) {
ro := &req.RequestOptions{ ro := &req.RequestOptions{
Params: map[string]string{"filter": "disabled"}, Params: map[string]string{"filter": "disabled"},
} }
res, err := a.c.baseRequest(http.MethodGet, routes.apps, ro) res, err := a.c.baseOcsRequest(http.MethodGet, routes.apps, ro)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -52,7 +52,7 @@ func (a *apps) ListDisabled() ([]string, error) {
//Infos return the app's details //Infos return the app's details
func (a *apps) Infos(name string) (App, error) { func (a *apps) Infos(name string) (App, error) {
res, err := a.c.baseRequest(http.MethodGet, routes.apps, nil, name) res, err := a.c.baseOcsRequest(http.MethodGet, routes.apps, nil, name)
if err != nil { if err != nil {
return App{}, err return App{}, err
} }
@@ -63,12 +63,12 @@ func (a *apps) Infos(name string) (App, error) {
//Enable enables an app //Enable enables an app
func (a *apps) Enable(name string) error { func (a *apps) Enable(name string) error {
_, err := a.c.baseRequest(http.MethodPost, routes.apps, nil, name) _, err := a.c.baseOcsRequest(http.MethodPost, routes.apps, nil, name)
return err return err
} }
//Disable disables an app //Disable disables an app
func (a *apps) Disable(name string) error { func (a *apps) Disable(name string) error {
_, err := a.c.baseRequest(http.MethodDelete, routes.apps, nil, name) _, err := a.c.baseOcsRequest(http.MethodDelete, routes.apps, nil, name)
return err return err
} }

View File

@@ -14,7 +14,7 @@ type appsConfig struct {
//List lists all the available apps //List lists all the available apps
func (a *appsConfig) List() (apps []string, err error) { func (a *appsConfig) List() (apps []string, err error) {
res, err := a.c.baseRequest(http.MethodGet, routes.appsConfig, nil) res, err := a.c.baseOcsRequest(http.MethodGet, routes.appsConfig, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -25,7 +25,7 @@ func (a *appsConfig) List() (apps []string, err error) {
//Keys returns the app's config keys //Keys returns the app's config keys
func (a *appsConfig) Keys(id string) (keys []string, err error) { func (a *appsConfig) Keys(id string) (keys []string, err error) {
res, err := a.c.baseRequest(http.MethodGet, routes.appsConfig, nil, id) res, err := a.c.baseOcsRequest(http.MethodGet, routes.appsConfig, nil, id)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -36,7 +36,7 @@ func (a *appsConfig) Keys(id string) (keys []string, err error) {
//Value get the config value for the given app's key //Value get the config value for the given app's key
func (a *appsConfig) Value(id, key string) (string, error) { func (a *appsConfig) Value(id, key string) (string, error) {
res, err := a.c.baseRequest(http.MethodGet, routes.appsConfig, nil, id, key) res, err := a.c.baseOcsRequest(http.MethodGet, routes.appsConfig, nil, id, key)
if err != nil { if err != nil {
return "", err return "", err
} }
@@ -52,13 +52,13 @@ func (a *appsConfig) SetValue(id, key, value string) error {
"value": value, "value": value,
}, },
} }
_, err := a.c.baseRequest(http.MethodPost, routes.appsConfig, ro, id, key) _, err := a.c.baseOcsRequest(http.MethodPost, routes.appsConfig, ro, id, key)
return err return err
} }
//DeleteValue delete the config value and (!! be careful !!) the key //DeleteValue delete the config value and (!! be careful !!) the key
func (a *appsConfig) DeleteValue(id, key, value string) error { func (a *appsConfig) DeleteValue(id, key, value string) error {
_, err := a.c.baseRequest(http.MethodDelete, routes.appsConfig, nil, id, key) _, err := a.c.baseOcsRequest(http.MethodDelete, routes.appsConfig, nil, id, key)
return err return err
} }

View File

@@ -1,6 +1,6 @@
package gonextcloud package gonextcloud
//Capabilities // Capabilities is the capabilities provided by the Nextcloud server
type Capabilities struct { type Capabilities struct {
Core struct { Core struct {
Pollinterval int `json:"pollinterval"` Pollinterval int `json:"pollinterval"`

View File

@@ -25,6 +25,7 @@ type client struct {
users *users users *users
groups *groups groups *groups
webdav *webDav webdav *webDav
passwords *passwords
} }
// newClient create a new client from the Nextcloud Instance URL // newClient create a new client from the Nextcloud Instance URL
@@ -52,6 +53,7 @@ func newClient(hostname string) (*client, error) {
c.shares = &shares{c} c.shares = &shares{c}
c.users = &users{c} c.users = &users{c}
c.groups = &groups{c} c.groups = &groups{c}
c.passwords = &passwords{c}
// Create empty webdav client // Create empty webdav client
// It will be replaced after login // It will be replaced after login
c.webdav = &webDav{Client: &gowebdav.Client{}} c.webdav = &webDav{Client: &gowebdav.Client{}}
@@ -97,3 +99,8 @@ func (c *client) Groups() Groups {
func (c *client) WebDav() WebDav { func (c *client) WebDav() WebDav {
return c.webdav return c.webdav
} }
// Password return the Password client Interface
func (c *client) Passwords() Passwords {
return c.passwords
}

4
doc.go
View File

@@ -1,5 +1,5 @@
/* /*
A simple Go Client for Nextcloud's API. Package gonextcloud is a simple Go Client for Nextcloud's API.
For more information about the Provisioning API, see the documentation: For more information about the Provisioning API, see the documentation:
https://docs.nextcloud.com/server/13/admin_manual/configuration_user/user_provisioning_api.html https://docs.nextcloud.com/server/13/admin_manual/configuration_user/user_provisioning_api.html
@@ -30,7 +30,7 @@ For example, to list all the Nextcloud's instance users:
} }
defer c.Logout() defer c.Logout()
users, err := c.users().List() users, err := c.Users().List()
if err != nil { if err != nil {
panic(err) panic(err)
} }

View File

@@ -5,7 +5,7 @@ import (
"strings" "strings"
) )
//APIError contains the returned error code and message from the Nextcloud's API // APIError contains the returned error code and message from the Nextcloud's API
type APIError struct { type APIError struct {
Code int Code int
Message string Message string
@@ -19,18 +19,18 @@ func errorFromMeta(meta meta) *APIError {
} }
} }
//Error return the types.APIError string // Error return the types.APIError string
func (e *APIError) Error() string { func (e *APIError) Error() string {
return fmt.Sprintf("%d : %s", e.Code, e.Message) return fmt.Sprintf("%d : %s", e.Code, e.Message)
} }
//UpdateError contains the user's field and corresponding error // UpdateError contains the user's field and corresponding error
type UpdateError struct { type UpdateError struct {
Field string Field string
Error error Error error
} }
//UpdateError contains the errors resulting from a UserUpdate or a UserCreateFull call // UserUpdateError contains the errors resulting from a UserUpdate or a UserCreateFull call
type UserUpdateError struct { type UserUpdateError struct {
Errors map[string]error Errors map[string]error
} }

View File

@@ -13,19 +13,33 @@ func NewClient(hostname string) (Client, error) {
// Client is the main client interface // Client is the main client interface
type Client interface { type Client interface {
// Nextcloud Apps client
Apps() Apps Apps() Apps
// Nextcloud App Config client
AppsConfig() AppsConfig AppsConfig() AppsConfig
// Nextcloud Group Folders client
GroupFolders() GroupFolders GroupFolders() GroupFolders
// Nextcloud Notifications client
Notifications() Notifications Notifications() Notifications
// Nextcloud Shares client
Shares() Shares Shares() Shares
// Nextcloud Users client
Users() Users Users() Users
// Nextcloud Groups client
Groups() Groups Groups() Groups
// Nextcloud WebDav (files) client
WebDav() WebDav WebDav() WebDav
// Nextcloud Monitoring client
Monitoring() (*Monitoring, error) Monitoring() (*Monitoring, error)
// Nextcloud Password app client
Passwords() Passwords
// Login authorize client
Login(username string, password string) error Login(username string, password string) error
// Logout clear connetion and session
Logout() error Logout() error
} }
// Auth is the standard auth methods
type Auth interface { type Auth interface {
Login(username string, password string) error Login(username string, password string) error
Logout() error Logout() error
@@ -168,3 +182,7 @@ type WebDav interface {
// directory in the tree, including root. // directory in the tree, including root.
Walk(path string, walkFunc filepath.WalkFunc) error Walk(path string, walkFunc filepath.WalkFunc) error
} }
type Passwords interface {
List() ([]Password, error)
}

View File

@@ -380,7 +380,7 @@ var (
"TestInvalidBaseRequest", "TestInvalidBaseRequest",
func(t *testing.T) { func(t *testing.T) {
c.baseURL = &url.URL{} c.baseURL = &url.URL{}
_, err := c.baseRequest(http.MethodGet, routes.capabilities, nil, "admin", "invalid") _, err := c.baseOcsRequest(http.MethodGet, routes.capabilities, nil, "admin", "invalid")
c = nil c = nil
assert.Error(t, err) assert.Error(t, err)
}, },
@@ -424,7 +424,7 @@ var (
"TestBaseRequest", "TestBaseRequest",
func(t *testing.T) { func(t *testing.T) {
c, _ = newClient("") c, _ = newClient("")
_, err := c.baseRequest(http.MethodGet, routes.capabilities, nil, "admin", "invalid") _, err := c.baseOcsRequest(http.MethodGet, routes.capabilities, nil, "admin", "invalid")
assert.Error(t, err) assert.Error(t, err)
}, },
}, },

View File

@@ -1,6 +1,6 @@
package gonextcloud package gonextcloud
//Group // Group is a Nextcloud group
type Group struct { type Group struct {
ID string `json:"id"` ID string `json:"id"`
Displayname string `json:"displayname"` Displayname string `json:"displayname"`

View File

@@ -18,6 +18,7 @@ type groupFolderBadFormatGroups struct {
Size int `json:"size"` Size int `json:"size"`
} }
// GroupFolder is group shared folder from groupfolders application
type GroupFolder struct { type GroupFolder struct {
ID int `json:"id"` ID int `json:"id"`
MountPoint string `json:"mount_point"` MountPoint string `json:"mount_point"`

View File

@@ -15,7 +15,7 @@ type groupFolders struct {
//List returns the groups folders //List returns the groups folders
func (g *groupFolders) List() (map[int]GroupFolder, error) { func (g *groupFolders) List() (map[int]GroupFolder, error) {
res, err := g.c.baseRequest(http.MethodGet, routes.groupfolders, nil) res, err := g.c.baseOcsRequest(http.MethodGet, routes.groupfolders, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -27,7 +27,7 @@ func (g *groupFolders) List() (map[int]GroupFolder, error) {
//Get returns the group folder details //Get returns the group folder details
func (g *groupFolders) Get(id int) (GroupFolder, error) { func (g *groupFolders) Get(id int) (GroupFolder, error) {
res, err := g.c.baseRequest(http.MethodGet, routes.groupfolders, nil, strconv.Itoa(id)) res, err := g.c.baseOcsRequest(http.MethodGet, routes.groupfolders, nil, strconv.Itoa(id))
if err != nil { if err != nil {
return GroupFolder{}, err return GroupFolder{}, err
} }
@@ -47,7 +47,7 @@ func (g *groupFolders) Create(name string) (id int, err error) {
"mountpoint": name, "mountpoint": name,
}, },
} }
res, err := g.c.baseRequest(http.MethodPost, routes.groupfolders, ro) res, err := g.c.baseOcsRequest(http.MethodPost, routes.groupfolders, ro)
if err != nil { if err != nil {
return 0, err return 0, err
} }
@@ -65,7 +65,7 @@ func (g *groupFolders) Rename(groupID int, name string) error {
}, },
} }
// groupFolders's response does not give any clues about success or failure // groupFolders's response does not give any clues about success or failure
_, err := g.c.baseRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(groupID), "mountpoint") _, err := g.c.baseOcsRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(groupID), "mountpoint")
if err != nil { if err != nil {
return err return err
} }
@@ -82,7 +82,7 @@ func (g *groupFolders) AddGroup(folderID int, groupName string) error {
}, },
} }
// groupFolders's response does not give any clues about success or failure // groupFolders's response does not give any clues about success or failure
_, err := g.c.baseRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(folderID), "groups") _, err := g.c.baseOcsRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(folderID), "groups")
if err != nil { if err != nil {
return err return err
} }
@@ -92,7 +92,7 @@ func (g *groupFolders) AddGroup(folderID int, groupName string) error {
//RemoveGroup remove a group from the group folder //RemoveGroup remove a group from the group folder
func (g *groupFolders) RemoveGroup(folderID int, groupName string) error { func (g *groupFolders) RemoveGroup(folderID int, groupName string) error {
// groupFolders's response does not give any clues about success or failure // groupFolders's response does not give any clues about success or failure
_, err := g.c.baseRequest(http.MethodDelete, routes.groupfolders, nil, strconv.Itoa(folderID), "groups", groupName) _, err := g.c.baseOcsRequest(http.MethodDelete, routes.groupfolders, nil, strconv.Itoa(folderID), "groups", groupName)
if err != nil { if err != nil {
return err return err
} }
@@ -107,7 +107,7 @@ func (g *groupFolders) SetGroupPermissions(folderID int, groupName string, permi
}, },
} }
// groupFolders's response does not give any clues about success or failure // groupFolders's response does not give any clues about success or failure
_, err := g.c.baseRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(folderID), "groups", groupName) _, err := g.c.baseOcsRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(folderID), "groups", groupName)
if err != nil { if err != nil {
return err return err
} }
@@ -122,7 +122,7 @@ func (g *groupFolders) SetQuota(folderID int, quota int) error {
}, },
} }
// groupFolders's response does not give any clues about success or failure // groupFolders's response does not give any clues about success or failure
_, err := g.c.baseRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(folderID), "quota") _, err := g.c.baseOcsRequest(http.MethodPost, routes.groupfolders, ro, strconv.Itoa(folderID), "quota")
if err != nil { if err != nil {
return err return err
} }

View File

@@ -13,7 +13,7 @@ type groups struct {
//List lists the Nextcloud groups //List lists the Nextcloud groups
func (g *groups) List() ([]string, error) { func (g *groups) List() ([]string, error) {
res, err := g.c.baseRequest(http.MethodGet, routes.groups, nil) res, err := g.c.baseOcsRequest(http.MethodGet, routes.groups, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -29,7 +29,7 @@ func (g *groups) ListDetails(search string) ([]Group, error) {
"search": search, "search": search,
}, },
} }
res, err := g.c.baseRequest(http.MethodGet, routes.groups, ro, "details") res, err := g.c.baseOcsRequest(http.MethodGet, routes.groups, ro, "details")
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -40,7 +40,7 @@ func (g *groups) ListDetails(search string) ([]Group, error) {
//users list the group's users //users list the group's users
func (g *groups) Users(name string) ([]string, error) { func (g *groups) Users(name string) ([]string, error) {
res, err := g.c.baseRequest(http.MethodGet, routes.groups, nil, name) res, err := g.c.baseOcsRequest(http.MethodGet, routes.groups, nil, name)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -54,7 +54,7 @@ func (g *groups) Search(search string) ([]string, error) {
ro := &req.RequestOptions{ ro := &req.RequestOptions{
Params: map[string]string{"search": search}, Params: map[string]string{"search": search},
} }
res, err := g.c.baseRequest(http.MethodGet, routes.groups, ro) res, err := g.c.baseOcsRequest(http.MethodGet, routes.groups, ro)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -80,7 +80,7 @@ func (g *groups) Delete(name string) error {
//SubAdminList lists the group's subadmins //SubAdminList lists the group's subadmins
func (g *groups) SubAdminList(name string) ([]string, error) { func (g *groups) SubAdminList(name string) ([]string, error) {
res, err := g.c.baseRequest(http.MethodGet, routes.groups, nil, name, "subadmins") res, err := g.c.baseOcsRequest(http.MethodGet, routes.groups, nil, name, "subadmins")
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -90,6 +90,6 @@ func (g *groups) SubAdminList(name string) ([]string, error) {
} }
func (g *groups) baseRequest(method string, ro *req.RequestOptions, subRoute ...string) error { func (g *groups) baseRequest(method string, ro *req.RequestOptions, subRoute ...string) error {
_, err := g.c.baseRequest(method, routes.groups, ro, subRoute...) _, err := g.c.baseOcsRequest(method, routes.groups, ro, subRoute...)
return err return err
} }

View File

@@ -1,5 +1,6 @@
package gonextcloud package gonextcloud
// System contains the operating system statistics
type System struct { type System struct {
Version string `json:"version"` Version string `json:"version"`
Theme string `json:"theme"` Theme string `json:"theme"`
@@ -18,7 +19,9 @@ type System struct {
SwapFree int `json:"swap_free"` SwapFree int `json:"swap_free"`
} }
// Monitoring contains the nextcloud monitoring statistics
type Monitoring struct { type Monitoring struct {
// Nextcloud Statistics
Nextcloud struct { Nextcloud struct {
System System `json:"system"` System System `json:"system"`
Storage Storage `json:"storage"` Storage Storage `json:"storage"`
@@ -32,6 +35,7 @@ type Monitoring struct {
NumFedSharesReceived int `json:"num_fed_shares_received"` NumFedSharesReceived int `json:"num_fed_shares_received"`
} `json:"shares"` } `json:"shares"`
} `json:"nextcloud"` } `json:"nextcloud"`
// Server statistics
Server struct { Server struct {
Webserver string `json:"webserver"` Webserver string `json:"webserver"`
Php struct { Php struct {
@@ -46,15 +50,18 @@ type Monitoring struct {
Size int `json:"size"` Size int `json:"size"`
} `json:"database"` } `json:"database"`
} `json:"server"` } `json:"server"`
// Active users statistics
ActiveUsers ActiveUsers `json:"activeUsers"` ActiveUsers ActiveUsers `json:"activeUsers"`
} }
// ActiveUsers contains the active users statistics
type ActiveUsers struct { type ActiveUsers struct {
Last5Minutes int `json:"last5minutes"` Last5Minutes int `json:"last5minutes"`
Last1Hour int `json:"last1hour"` Last1Hour int `json:"last1hour"`
Last24Hours int `json:"last24hours"` Last24Hours int `json:"last24hours"`
} }
// Storage contains the storage statistics
type Storage struct { type Storage struct {
NumUsers int `json:"num_users"` NumUsers int `json:"num_users"`
NumFiles int `json:"num_files"` NumFiles int `json:"num_files"`

View File

@@ -6,7 +6,7 @@ import (
//Monitoring return nextcloud monitoring statistics //Monitoring return nextcloud monitoring statistics
func (c *client) Monitoring() (*Monitoring, error) { func (c *client) Monitoring() (*Monitoring, error) {
res, err := c.baseRequest(http.MethodGet, routes.monitor, nil) res, err := c.baseOcsRequest(http.MethodGet, routes.monitor, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }

View File

@@ -2,6 +2,7 @@ package gonextcloud
import "time" import "time"
// Notification is a nextcloud notification (from notification app)
type Notification struct { type Notification struct {
NotificationID int `json:"notification_id"` NotificationID int `json:"notification_id"`
App string `json:"app"` App string `json:"app"`

View File

@@ -18,7 +18,7 @@ func (n *notifications) List() ([]Notification, error) {
if err := n.Available(); err != nil { if err := n.Available(); err != nil {
return nil, err return nil, err
} }
res, err := n.c.baseRequest(http.MethodGet, routes.notifications, nil) res, err := n.c.baseOcsRequest(http.MethodGet, routes.notifications, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -32,7 +32,7 @@ func (n *notifications) Get(id int) (Notification, error) {
if err := n.Available(); err != nil { if err := n.Available(); err != nil {
return Notification{}, err return Notification{}, err
} }
res, err := n.c.baseRequest(http.MethodGet, routes.notifications, nil, strconv.Itoa(id)) res, err := n.c.baseOcsRequest(http.MethodGet, routes.notifications, nil, strconv.Itoa(id))
if err != nil { if err != nil {
return Notification{}, err return Notification{}, err
} }
@@ -46,7 +46,7 @@ func (n *notifications) Delete(id int) error {
if err := n.Available(); err != nil { if err := n.Available(); err != nil {
return err return err
} }
_, err := n.c.baseRequest(http.MethodDelete, routes.notifications, nil, strconv.Itoa(id)) _, err := n.c.baseOcsRequest(http.MethodDelete, routes.notifications, nil, strconv.Itoa(id))
return err return err
} }
@@ -55,7 +55,7 @@ func (n *notifications) DeleteAll() error {
if err := n.Available(); err != nil { if err := n.Available(); err != nil {
return err return err
} }
_, err := n.c.baseRequest(http.MethodDelete, routes.notifications, nil) _, err := n.c.baseOcsRequest(http.MethodDelete, routes.notifications, nil)
return err return err
} }
@@ -70,7 +70,7 @@ func (n *notifications) Create(userID, title, message string) error {
"longMessage": message, "longMessage": message,
}, },
} }
_, err := n.c.baseRequest(http.MethodPost, routes.adminNotifications, ro, userID) _, err := n.c.baseOcsRequest(http.MethodPost, routes.adminNotifications, ro, userID)
return err return err
} }

62
password_impl.go Normal file
View File

@@ -0,0 +1,62 @@
package gonextcloud
import (
"net/http"
"strconv"
"time"
)
//passwords contains some passwords app available actions
type passwords struct {
c *client
}
func (p *passwords) List() ([]Password, error) {
res, err := p.c.baseRequest(http.MethodGet, routes.passwords, nil)
if err != nil {
return nil, err
}
var r []Password
err = res.JSON(&r)
if err != nil {
return nil, err
}
return r, nil
}
type passwordTime struct {
time.Time
}
func (pt *passwordTime) UnmarshalJSON(b []byte) (err error) {
s := string(b)
i, err := strconv.ParseInt(s, 10, 64)
if err != nil {
return err
}
t := time.Unix(i, 0)
pt.Time = t
return
}
type Password struct {
Id string `json:"id"`
Label string `json:"label"`
Username string `json:"username"`
Password string `json:"password"`
Url string `json:"url"`
Notes string `json:"notes"`
Status int `json:"status"`
StatusCode string `json:"statusCode"`
Hash string `json:"hash"`
Folder string `json:"foler"`
Revision string `json:"revision"`
Share string `json:"share"`
CseType string `json:"cseType"`
SseType string `json:"ssetype"`
Favorite bool `json:"favorite"`
Editable bool `json:"editable"`
Edited passwordTime `json:"edited"`
Created passwordTime `json:"created"`
Updated passwordTime `json:"updated"`
}

View File

@@ -119,6 +119,7 @@ type capabilitiesResponse struct {
} `json:"ocs"` } `json:"ocs"`
} }
// Version contains the nextcloud version informations
type Version struct { type Version struct {
Major int `json:"major"` Major int `json:"major"`
Minor int `json:"minor"` Minor int `json:"minor"`

View File

@@ -14,6 +14,7 @@ type apiRoutes struct {
appsConfig *url.URL appsConfig *url.URL
notifications *url.URL notifications *url.URL
adminNotifications *url.URL adminNotifications *url.URL
passwords *url.URL
} }
const badRequest = 998 const badRequest = 998
@@ -31,5 +32,6 @@ var (
appsConfig: &url.URL{Path: apiPath.Path + "/apps/provisioning_api/api/v1/config/apps"}, appsConfig: &url.URL{Path: apiPath.Path + "/apps/provisioning_api/api/v1/config/apps"},
notifications: &url.URL{Path: apiPath.Path + "/apps/notifications/api/v2/notifications"}, notifications: &url.URL{Path: apiPath.Path + "/apps/notifications/api/v2/notifications"},
adminNotifications: &url.URL{Path: apiPath.Path + "/apps/admin_notifications/api/v2/notifications"}, adminNotifications: &url.URL{Path: apiPath.Path + "/apps/admin_notifications/api/v2/notifications"},
passwords: &url.URL{Path: "/index.php/apps/passwords/api/1.0/password/list"},
} }
) )

View File

@@ -1,22 +1,36 @@
package gonextcloud package gonextcloud
// ShareType is the nextcloud shares types enum :
type ShareType int type ShareType int
// SharePermission is the nextcloud share permissions enum
type SharePermission int type SharePermission int
const ( const (
UserShare ShareType = 0 // UserShare is a file or folder shared with other user(s)
GroupShare ShareType = 1 UserShare ShareType = 0
PublicLinkShare ShareType = 3 // GroupShare is a file or folder shared with a group
GroupShare ShareType = 1
// PublicLinkShare is a file or folder shared through public link
PublicLinkShare ShareType = 3
// FederatedCloudShare is a file or folder shared through federated cloud
FederatedCloudShare ShareType = 6 FederatedCloudShare ShareType = 6
ReadPermission SharePermission = 1 // ReadPermission grant read permission
UpdatePermission SharePermission = 2 ReadPermission SharePermission = 1
CreatePermission SharePermission = 4 // UpdatePermission grant update permission
DeletePermission SharePermission = 8 UpdatePermission SharePermission = 2
// CreatePermission grant create permission
CreatePermission SharePermission = 4
// DeletePermission grant delete permission
DeletePermission SharePermission = 8
// ReSharePermission grant resharing permission
ReSharePermission SharePermission = 16 ReSharePermission SharePermission = 16
AllPermissions SharePermission = 31 // AllPermissions grant all permissions
AllPermissions SharePermission = 31
) )
// ShareUpdate contains the data required in order to update a nextcloud share
type ShareUpdate struct { type ShareUpdate struct {
ShareID int ShareID int
Permissions SharePermission Permissions SharePermission
@@ -25,6 +39,7 @@ type ShareUpdate struct {
ExpireDate string ExpireDate string
} }
// Share is a nextcloud share
type Share struct { type Share struct {
ID string `json:"id"` ID string `json:"id"`
ShareType int `json:"share_type"` ShareType int `json:"share_type"`

View File

@@ -16,7 +16,7 @@ type shares struct {
//List list all shares of the logged in user //List list all shares of the logged in user
func (s *shares) List() ([]Share, error) { func (s *shares) List() ([]Share, error) {
res, err := s.c.baseRequest(http.MethodGet, routes.shares, nil) res, err := s.c.baseOcsRequest(http.MethodGet, routes.shares, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -34,7 +34,7 @@ func (s *shares) GetFromPath(path string, reshares bool, subfiles bool) ([]Share
"subfiles": strconv.FormatBool(subfiles), "subfiles": strconv.FormatBool(subfiles),
}, },
} }
res, err := s.c.baseRequest(http.MethodGet, routes.shares, ro) res, err := s.c.baseOcsRequest(http.MethodGet, routes.shares, ro)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -45,7 +45,7 @@ func (s *shares) GetFromPath(path string, reshares bool, subfiles bool) ([]Share
//Get information about a known Share //Get information about a known Share
func (s *shares) Get(shareID string) (Share, error) { func (s *shares) Get(shareID string) (Share, error) {
res, err := s.c.baseRequest(http.MethodGet, routes.shares, nil, shareID) res, err := s.c.baseOcsRequest(http.MethodGet, routes.shares, nil, shareID)
if err != nil { if err != nil {
return Share{}, err return Share{}, err
} }
@@ -77,7 +77,7 @@ func (s *shares) Create(
"permissions": strconv.Itoa(int(permission)), "permissions": strconv.Itoa(int(permission)),
}, },
} }
res, err := s.c.baseRequest(http.MethodPost, routes.shares, ro) res, err := s.c.baseOcsRequest(http.MethodPost, routes.shares, ro)
if err != nil { if err != nil {
return Share{}, err return Share{}, err
} }
@@ -88,7 +88,7 @@ func (s *shares) Create(
//Delete Remove the given share. //Delete Remove the given share.
func (s *shares) Delete(shareID int) error { func (s *shares) Delete(shareID int) error {
_, err := s.c.baseRequest(http.MethodDelete, routes.shares, nil, strconv.Itoa(shareID)) _, err := s.c.baseOcsRequest(http.MethodDelete, routes.shares, nil, strconv.Itoa(shareID))
return err return err
} }
@@ -169,6 +169,6 @@ func (s *shares) baseShareUpdate(shareID string, key string, value string) error
ro := &req.RequestOptions{ ro := &req.RequestOptions{
Data: map[string]string{key: value}, Data: map[string]string{key: value},
} }
_, err := s.c.baseRequest(http.MethodPut, routes.shares, ro, shareID) _, err := s.c.baseOcsRequest(http.MethodPut, routes.shares, ro, shareID)
return err return err
} }

View File

@@ -2,7 +2,7 @@ package gonextcloud
import "strconv" import "strconv"
//User encapsulate the data needed to create a new Nextcloud's User // User encapsulate the data needed to create a new Nextcloud's User
type User struct { type User struct {
Username string Username string
Email string Email string
@@ -12,7 +12,7 @@ type User struct {
Groups []string Groups []string
} }
//UserDetails is the raw Nextcloud User response // UserDetails is the raw Nextcloud User response
type UserDetails struct { type UserDetails struct {
Enabled bool `json:"enabled"` Enabled bool `json:"enabled"`
ID string `json:"id"` ID string `json:"id"`
@@ -33,6 +33,7 @@ type UserDetails struct {
Locale string `json:"locale,omitempty"` Locale string `json:"locale,omitempty"`
} }
// Quota is a use storage Quota
type Quota struct { type Quota struct {
Free int64 `json:"free"` Free int64 `json:"free"`
Used int64 `json:"used"` Used int64 `json:"used"`

View File

@@ -20,7 +20,7 @@ type users struct {
// List return the Nextcloud'user list // List return the Nextcloud'user list
func (u *users) List() ([]string, error) { func (u *users) List() ([]string, error) {
res, err := u.c.baseRequest(http.MethodGet, routes.users, nil) res, err := u.c.baseOcsRequest(http.MethodGet, routes.users, nil)
//res, err := c.session.Get(u.String(), nil) //res, err := c.session.Get(u.String(), nil)
if err != nil { if err != nil {
return nil, err return nil, err
@@ -32,7 +32,7 @@ func (u *users) List() ([]string, error) {
//ListDetails return a map of user with details //ListDetails return a map of user with details
func (u *users) ListDetails() (map[string]UserDetails, error) { func (u *users) ListDetails() (map[string]UserDetails, error) {
res, err := u.c.baseRequest(http.MethodGet, routes.users, nil, "details") res, err := u.c.baseOcsRequest(http.MethodGet, routes.users, nil, "details")
//res, err := c.session.Get(u.String(), nil) //res, err := c.session.Get(u.String(), nil)
if err != nil { if err != nil {
return nil, err return nil, err
@@ -47,7 +47,7 @@ func (u *users) Get(name string) (*UserDetails, error) {
if name == "" { if name == "" {
return nil, &APIError{Message: "name cannot be empty"} return nil, &APIError{Message: "name cannot be empty"}
} }
res, err := u.c.baseRequest(http.MethodGet, routes.users, nil, name) res, err := u.c.baseOcsRequest(http.MethodGet, routes.users, nil, name)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -66,7 +66,7 @@ func (u *users) Search(search string) ([]string, error) {
ro := &req.RequestOptions{ ro := &req.RequestOptions{
Params: map[string]string{"search": search}, Params: map[string]string{"search": search},
} }
res, err := u.c.baseRequest(http.MethodGet, routes.users, ro) res, err := u.c.baseOcsRequest(http.MethodGet, routes.users, ro)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -339,7 +339,7 @@ func (u *users) UpdateQuota(name string, quota int64) error {
//GroupList lists the user's groups //GroupList lists the user's groups
func (u *users) GroupList(name string) ([]string, error) { func (u *users) GroupList(name string) ([]string, error) {
res, err := u.c.baseRequest(http.MethodGet, routes.users, nil, name, "groups") res, err := u.c.baseOcsRequest(http.MethodGet, routes.users, nil, name, "groups")
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -415,7 +415,7 @@ func (u *users) updateAttribute(name string, key string, value string) error {
} }
func (u *users) baseRequest(method string, ro *req.RequestOptions, subRoutes ...string) error { func (u *users) baseRequest(method string, ro *req.RequestOptions, subRoutes ...string) error {
_, err := u.c.baseRequest(method, routes.users, ro, subRoutes...) _, err := u.c.baseOcsRequest(method, routes.users, ro, subRoutes...)
return err return err
} }

View File

@@ -10,6 +10,22 @@ import (
req "github.com/levigross/grequests" req "github.com/levigross/grequests"
) )
func (c *client) baseOcsRequest(method string, route *url.URL, ro *req.RequestOptions, subRoutes ...string) (*req.Response, error) {
res, err := c.baseRequest(method, route, ro, subRoutes...)
if err != nil {
return nil, err
}
// As we cannot read the ReaderCloser twice, we use the string content
js := res.String()
var r baseResponse
json.Unmarshal([]byte(js), &r)
if r.Ocs.Meta.Statuscode == 200 || r.Ocs.Meta.Statuscode == 100 {
return res, nil
}
err = errorFromMeta(r.Ocs.Meta)
return nil, err
}
func (c *client) baseRequest(method string, route *url.URL, ro *req.RequestOptions, subRoutes ...string) (*req.Response, error) { func (c *client) baseRequest(method string, route *url.URL, ro *req.RequestOptions, subRoutes ...string) (*req.Response, error) {
if !c.loggedIn() { if !c.loggedIn() {
return nil, errUnauthorized return nil, errUnauthorized
@@ -38,15 +54,7 @@ func (c *client) baseRequest(method string, route *url.URL, ro *req.RequestOptio
if err != nil { if err != nil {
return nil, err return nil, err
} }
// As we cannot read the ReaderCloser twice, we use the string content return res, nil
js := res.String()
var r baseResponse
json.Unmarshal([]byte(js), &r)
if r.Ocs.Meta.Statuscode == 200 || r.Ocs.Meta.Statuscode == 100 {
return res, nil
}
err = errorFromMeta(r.Ocs.Meta)
return nil, err
} }
func reformatJSON(json string) string { func reformatJSON(json string) string {