add accntcli

This commit is contained in:
2026-03-07 19:11:20 +02:00
parent 46bcc465ee
commit efdabf3efc
23 changed files with 698 additions and 96 deletions
+5 -2
View File
@@ -13,11 +13,14 @@ type SendHelloParams struct{}
type SendHelloResult struct {
Message string `json:"message"`
Alive bool `json:"alive"`
}
func (oper *Operator) SendHello(param *SendHelloParams) (*SendHelloResult, error) {
var err error
res := &SendHelloResult{}
res.Message = "hello"
res := &SendHelloResult{
Alive: true,
Message: "hello",
}
return res, err
}
+235
View File
@@ -0,0 +1,235 @@
/*
* Copyright 2026 Oleg Borodin <onborodin@gmail.com>
*
* This work is published and licensed under a Creative Commons
* Attribution-NonCommercial-NoDerivatives 4.0 International License.
*
* Distribution of this work is permitted, but commercial use and
* modifications are strictly prohibited.
*/
package accntcli
import (
"context"
"encoding/json"
"errors"
"mstore/app/accoper"
"mstore/app/handler"
"mstore/pkg/descr"
)
func (cli *Client) CreateAccount(ctx context.Context, host, user, pass string) (string, error) {
var err error
var res string
params := accoper.CreateAccountParams{
Username: user,
Password: pass,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
respdata, err := cli.doHTTPCall(ctx, host, "accont", "create", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.CreateAccountResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.AccountID
return res, err
}
func (cli *Client) GetAccountByID(ctx context.Context, host, accountID string) (*descr.AccountShort, error) {
var err error
res := &descr.AccountShort{}
params := accoper.GetAccountParams{
AccountID: accountID,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
resdata, err := cli.doHTTPCall(ctx, host, "account", "get", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.GetAccountResult]()
err = json.Unmarshal(resdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.Account
return res, err
}
func (cli *Client) GetAccountByName(ctx context.Context, host, username string) (*descr.AccountShort, error) {
var err error
res := &descr.AccountShort{}
params := accoper.GetAccountParams{
Username: username,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
resdata, err := cli.doHTTPCall(ctx, host, "account", "get", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.GetAccountResult]()
err = json.Unmarshal(resdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.Account
return res, err
}
func (cli *Client) UpdateAccountByID(ctx context.Context, host string, accountID, newUsername, newPassword string) error {
var err error
params := accoper.UpdateAccountParams{
AccountID: accountID,
NewUsername: newUsername,
NewPassword: newPassword,
}
reqdata, err := json.Marshal(params)
if err != nil {
return err
}
resdata, err := cli.doHTTPCall(ctx, host, "account", "update", reqdata)
if err != nil {
return err
}
response := handler.NewResponse[accoper.UpdateAccountResult]()
err = json.Unmarshal(resdata, response)
if err != nil {
return err
}
if response.Error {
err = errors.New(response.Message)
return err
}
return err
}
func (cli *Client) UpdateAccountByName(ctx context.Context, host, username, newUsername, newPassword string) error {
var err error
params := accoper.UpdateAccountParams{
Username: username,
NewUsername: newUsername,
NewPassword: newPassword,
}
reqdata, err := json.Marshal(params)
if err != nil {
return err
}
resdata, err := cli.doHTTPCall(ctx, host, "account", "update", reqdata)
if err != nil {
return err
}
response := handler.NewResponse[accoper.UpdateAccountResult]()
err = json.Unmarshal(resdata, response)
if err != nil {
return err
}
if response.Error {
err = errors.New(response.Message)
return err
}
return err
}
func (cli *Client) DeleteAccountByName(ctx context.Context, host, username string) error {
var err error
params := accoper.DeleteAccountParams{
Username: username,
}
reqdata, err := json.Marshal(params)
if err != nil {
return err
}
resdata, err := cli.doHTTPCall(ctx, host, "account", "delete", reqdata)
if err != nil {
return err
}
response := handler.NewResponse[accoper.DeleteAccountResult]()
err = json.Unmarshal(resdata, response)
if err != nil {
return err
}
if response.Error {
err = errors.New(response.Message)
return err
}
return err
}
func (cli *Client) DeleteAccountByID(ctx context.Context, host, accountID string) error {
var err error
params := accoper.DeleteAccountParams{
AccountID: accountID,
}
reqdata, err := json.Marshal(params)
if err != nil {
return err
}
resdata, err := cli.doHTTPCall(ctx, host, "account", "delete", reqdata)
if err != nil {
return err
}
response := handler.NewResponse[accoper.DeleteAccountResult]()
err = json.Unmarshal(resdata, response)
if err != nil {
return err
}
if response.Error {
err = errors.New(response.Message)
return err
}
return err
}
func (cli *Client) ListAccounts(ctx context.Context, host string) ([]descr.AccountShort, error) {
var err error
res := make([]descr.AccountShort, 0)
params := accoper.ListAccountsParams{}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
resdata, err := cli.doHTTPCall(ctx, host, "accounts", "list", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.ListAccountsResult]()
err = json.Unmarshal(resdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.Accounts
return res, err
}
@@ -1,4 +1,4 @@
package rpccli
package accntcli
import (
"crypto/tls"
+216
View File
@@ -0,0 +1,216 @@
/*
* Copyright 2026 Oleg Borodin <onborodin@gmail.com>
*
* This work is published and licensed under a Creative Commons
* Attribution-NonCommercial-NoDerivatives 4.0 International License.
*
* Distribution of this work is permitted, but commercial use and
* modifications are strictly prohibited.
*/
package accntcli
import (
"context"
"encoding/json"
"errors"
"mstore/app/accoper"
"mstore/app/handler"
"mstore/pkg/descr"
)
func (cli *Client) CreateGrantByAccountID(ctx context.Context, host string, accountID, right, pattern string) (string, error) {
var err error
var res string
params := accoper.CreateGrantParams{
AccountID: accountID,
Right: right,
Pattern: pattern,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
respdata, err := cli.doHTTPCall(ctx, host, "grant", "create", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.CreateGrantResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.GrantID
return res, err
}
func (cli *Client) CreateGrantByUsername(ctx context.Context, host, username string, right string, pattern string) (string, error) {
var err error
var res string
params := accoper.CreateGrantParams{
Username: username,
Right: right,
Pattern: pattern,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
respdata, err := cli.doHTTPCall(ctx, host, "grant", "create", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.CreateGrantResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.GrantID
return res, err
}
func (cli *Client) GetGrant(ctx context.Context, host, grantID string) (*descr.Grant, error) {
var err error
res := &descr.Grant{}
params := accoper.GetGrantParams{
GrantID: grantID,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
respdata, err := cli.doHTTPCall(ctx, host, "grant", "get", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.GetGrantResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.Grant
return res, err
}
func (cli *Client) UpdateGrant(ctx context.Context, host, grantID, newPattern string) error {
var err error
params := accoper.UpdateGrantParams{
GrantID: grantID,
NewPattern: newPattern,
}
reqdata, err := json.Marshal(params)
if err != nil {
return err
}
respdata, err := cli.doHTTPCall(ctx, host, "grant", "update", reqdata)
if err != nil {
return err
}
response := handler.NewResponse[accoper.UpdateGrantResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return err
}
if response.Error {
err = errors.New(response.Message)
return err
}
return err
}
func (cli *Client) DeleteGrant(ctx context.Context, host, grantID string) error {
var err error
params := accoper.DeleteGrantParams{
GrantID: grantID,
}
reqdata, err := json.Marshal(params)
if err != nil {
return err
}
respdata, err := cli.doHTTPCall(ctx, host, "grant", "delete", reqdata)
if err != nil {
return err
}
response := handler.NewResponse[accoper.DeleteGrantResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return err
}
if response.Error {
err = errors.New(response.Message)
return err
}
return err
}
func (cli *Client) ListGrantsByAccountID(ctx context.Context, host, accountID string) ([]descr.Grant, error) {
var err error
res := make([]descr.Grant, 0)
params := accoper.ListGrantsParams{
AccountID: accountID,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
respdata, err := cli.doHTTPCall(ctx, host, "grants", "list", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.ListGrantsResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.Grants
return res, err
}
func (cli *Client) ListGrantsByUsername(ctx context.Context, host, username string) ([]descr.Grant, error) {
var err error
res := make([]descr.Grant, 0)
params := accoper.ListGrantsParams{
Username: username,
}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
respdata, err := cli.doHTTPCall(ctx, host, "grants", "list", reqdata)
if err != nil {
return res, err
}
response := handler.NewResponse[accoper.ListGrantsResult]()
err = json.Unmarshal(respdata, response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.Grants
return res, err
}
+54
View File
@@ -0,0 +1,54 @@
package accntcli
import (
"net/url"
"path"
"strings"
)
type Referer struct {
urlobj *url.URL
user, pass string
obj, oper string
}
func NewReferer(hostname, object, operation string) (*Referer, error) {
ref := &Referer{
obj: object,
oper: operation,
}
if !strings.Contains(hostname, "://") {
hostname = "https://" + hostname
}
urlobj, err := url.Parse(hostname)
if err != nil {
return ref, err
}
if urlobj.User != nil {
ref.user = urlobj.User.Username()
ref.pass, _ = urlobj.User.Password()
urlobj.User = nil
}
urlobj.Path = "/"
ref.urlobj = urlobj
return ref, err
}
func (ref *Referer) Raw() string {
return path.Join(ref.urlobj.Host, "/v3/api/", ref.obj, ref.oper)
}
func (ref *Referer) Point() string {
curl := ref.urlobj.JoinPath("/v3/api/", ref.obj, ref.oper)
return curl.String()
}
func (ref *Referer) Userinfo() (string, string) {
return ref.user, ref.pass
}
func (ref *Referer) SetUserinfo(user, pass string) {
if user != "" && pass != "" {
ref.user, ref.pass = user, pass
}
}
+93
View File
@@ -0,0 +1,93 @@
/*
* Copyright 2026 Oleg Borodin <onborodin@gmail.com>
*
* This work is published and licensed under a Creative Commons
* Attribution-NonCommercial-NoDerivatives 4.0 International License.
*
* Distribution of this work is permitted, but commercial use and
* modifications are strictly prohibited.
*/
package accntcli
import (
"bytes"
"context"
"encoding/json"
"errors"
"fmt"
"io"
"net/http"
"strconv"
"mstore/app/handler"
"mstore/app/imageoper"
)
func (cli *Client) ServiceHello(ctx context.Context, host string) (bool, error) {
var res bool
var err error
params := imageoper.SendHelloParams{}
reqdata, err := json.Marshal(params)
if err != nil {
return res, err
}
resdata, err := cli.doHTTPCall(ctx, host, "service", "hello", reqdata)
if err != nil {
return res, err
}
response := handler.Response[imageoper.SendHelloResult]{}
err = json.Unmarshal(resdata, &response)
if err != nil {
return res, err
}
if response.Error {
err = errors.New(response.Message)
return res, err
}
res = response.Result.Alive
return res, err
}
func (cli *Client) doHTTPCall(ctx context.Context, host, obj, oper string, req []byte) ([]byte, error) {
var err error
var res []byte
reader := bytes.NewReader(req)
ref, err := NewReferer(host, obj, oper)
if err != nil {
return res, err
}
httpReq, err := http.NewRequestWithContext(ctx, http.MethodPost, ref.Point(), reader)
if err != nil {
return res, err
}
httpReq.Header.Set("User-Agent", cli.userAgent)
httpReq.Header.Set("Accept", "*/*")
httpResp, err := cli.httpClient.Do(httpReq)
if err != nil {
return res, err
}
defer httpResp.Body.Close()
if httpResp.StatusCode != http.StatusOK {
err := fmt.Errorf("Unexpected response code: %s", httpResp.Status)
return res, err
}
contentLength := httpResp.Header.Get("Content-Length")
if contentLength == "" {
err := fmt.Errorf("Content-Length header is missing")
return res, err
}
blobSize, err := strconv.ParseInt(contentLength, 10, 64)
if err != nil {
return res, err
}
buffer := bytes.NewBuffer(nil)
recSize, err := io.Copy(buffer, httpResp.Body)
if blobSize != recSize {
err := fmt.Errorf("Mismatch declared and actual body size, %d and %d", blobSize, recSize)
return res, err
}
res = buffer.Bytes()
return res, err
}
+5 -1
View File
@@ -38,6 +38,10 @@ func (cli *Client) GetFile(ctx context.Context, rawpath string, writer io.Writer
return exist, err
}
contentLength := resp.Header.Get("Content-Length")
if contentLength == "" {
err := fmt.Errorf("Content-Length header is missing")
return exist, err
}
blobSize, err := strconv.ParseInt(contentLength, 10, 64)
if err != nil {
return exist, err
@@ -45,7 +49,7 @@ func (cli *Client) GetFile(ctx context.Context, rawpath string, writer io.Writer
recSize, err := Copy(ctx, writer, resp.Body)
if blobSize != recSize {
err := fmt.Errorf("Mismatch declared and actual body size, %d and %d", blobSize, recSize)
err := fmt.Errorf("Mismatch declared and actual body size: %d and %d", blobSize, recSize)
return exist, err
}
exist = true
+5 -1
View File
@@ -38,6 +38,10 @@ func (cli *Client) ListCollections(ctx context.Context, rawpath string) ([]byte,
return list, err
}
contentLength := resp.Header.Get("Content-Length")
if contentLength == "" {
err := fmt.Errorf("Content-Length header is missing")
return list, err
}
blobSize, err := strconv.ParseInt(contentLength, 10, 64)
if err != nil {
return list, err
@@ -45,7 +49,7 @@ func (cli *Client) ListCollections(ctx context.Context, rawpath string) ([]byte,
buffer := bytes.NewBuffer(nil)
recSize, err := Copy(ctx, buffer, resp.Body)
if blobSize != recSize {
err := fmt.Errorf("Mismatch declared and actual body size, %d and %d", blobSize, recSize)
err := fmt.Errorf("Mismatch declared and actual body size: %d and %d", blobSize, recSize)
return list, err
}
list = buffer.Bytes()
+5 -1
View File
@@ -38,6 +38,10 @@ func (cli *Client) ListFiles(ctx context.Context, rawpath string) ([]byte, error
return list, err
}
contentLength := resp.Header.Get("Content-Length")
if contentLength == "" {
err := fmt.Errorf("Content-Length header is missing")
return list, err
}
blobSize, err := strconv.ParseInt(contentLength, 10, 64)
if err != nil {
return list, err
@@ -45,7 +49,7 @@ func (cli *Client) ListFiles(ctx context.Context, rawpath string) ([]byte, error
buffer := bytes.NewBuffer(nil)
recSize, err := Copy(ctx, buffer, resp.Body)
if blobSize != recSize {
err := fmt.Errorf("Mismatch declared and actual body size, %d and %d", blobSize, recSize)
err := fmt.Errorf("Mismatch declared and actual body size: %d and %d", blobSize, recSize)
return list, err
}
list = buffer.Bytes()
+14 -14
View File
@@ -13,15 +13,15 @@ const (
PathTypeRegexp = "regexp"
)
type Repository struct {
type Referer struct {
urlobj *url.URL
user, pass string
resource string
values url.Values
}
func ParsePath(rawpath string) (*Repository, error) {
repo := &Repository{
func ParsePath(rawpath string) (*Referer, error) {
repo := &Referer{
values: url.Values{},
}
if !strings.Contains(rawpath, "://") {
@@ -43,7 +43,7 @@ func ParsePath(rawpath string) (*Repository, error) {
return repo, err
}
func (repo *Repository) Raw() string {
func (repo *Referer) Raw() string {
res := path.Join(repo.urlobj.Host, repo.resource)
query := repo.values.Encode()
if query != "" {
@@ -52,47 +52,47 @@ func (repo *Repository) Raw() string {
return res
}
func (repo *Repository) SetResource(resource string) {
func (repo *Referer) SetResource(resource string) {
repo.resource = path.Join("/", resource)
}
func (repo *Repository) JoinResource(resource string) {
func (repo *Referer) JoinResource(resource string) {
repo.resource = path.Join("/", repo.resource, resource)
}
func (repo *Repository) PathType(typ string) {
func (repo *Referer) PathType(typ string) {
repo.values.Set("pathType", typ)
}
func (repo *Repository) DryRun(yesno bool) {
func (repo *Referer) DryRun(yesno bool) {
repo.values.Set("dryRun", strconv.FormatBool(yesno))
}
func (repo *Repository) File() string {
func (repo *Referer) File() string {
curl := repo.urlobj.JoinPath("/v3/api/file/", repo.resource)
return curl.String()
}
func (repo *Repository) Files() string {
func (repo *Referer) Files() string {
curl := repo.urlobj.JoinPath("/v3/api/files/", repo.resource)
return curl.String()
}
func (repo *Repository) Collection() string {
func (repo *Referer) Collection() string {
curl := repo.urlobj.JoinPath("/v3/api/collection/", repo.resource)
return curl.String()
}
func (repo *Repository) Collections() string {
func (repo *Referer) Collections() string {
curl := repo.urlobj.JoinPath("/v3/api/collections/", repo.resource)
return curl.String()
}
func (repo *Repository) Userinfo() (string, string) {
func (repo *Referer) Userinfo() (string, string) {
return repo.user, repo.pass
}
func (repo *Repository) SetUserinfo(user, pass string) {
func (repo *Referer) SetUserinfo(user, pass string) {
if user != "" && pass != "" {
repo.user, repo.pass = user, pass
}
+1 -1
View File
@@ -12,7 +12,7 @@ func (cli *Client) BlobExists(ctx context.Context, rawrepo string, digest string
var exist bool
var size int64
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return exist, size, err
}
+1 -1
View File
@@ -10,7 +10,7 @@ func (cli *Client) DeleteBlob(ctx context.Context, rawrepo, digest string) (bool
var err error
var exist bool
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return exist, err
}
+1 -1
View File
@@ -10,7 +10,7 @@ func (cli *Client) DeleteManifest(ctx context.Context, rawrepo, tag string) (boo
var err error
var exist bool
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return exist, err
}
+1 -1
View File
@@ -12,7 +12,7 @@ func (cli *Client) GetBlob(ctx context.Context, rawrepo string, writer io.Writer
var err error
var exist bool
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return exist, err
}
+1 -1
View File
@@ -19,7 +19,7 @@ func (cli *Client) GetManifest(ctx context.Context, rawrepo, tag string) (bool,
var mime string
var man []byte
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return exist, mime, man, err
}
+1 -1
View File
@@ -10,7 +10,7 @@ func (cli *Client) GetUpload(ctx context.Context, rawrepo string) (string, error
var err error
var loc string
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return loc, err
}
+1 -1
View File
@@ -14,7 +14,7 @@ func (cli *Client) ManifestExists(ctx context.Context, rawrepo, tag string) (boo
var size int64
var csum string
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return exist, mime, size, csum, err
}
+45
View File
@@ -0,0 +1,45 @@
package repocli
import (
"context"
"fmt"
"io"
"net/http"
"strconv"
)
func (cli *Client) PatchUpload(ctx context.Context, rawrepo string, src io.Reader, uploc string, size int64) (string, error) {
var err error
var ouloc string
ref, err := NewReferer(rawrepo)
if err != nil {
return ouloc, err
}
uri, err := ref.Patch(uploc)
if err != nil {
return ouloc, err
}
req, err := http.NewRequestWithContext(ctx, http.MethodPatch, uri, src)
if err != nil {
return ouloc, err
}
req.Header.Set("User-Agent", cli.userAgent)
req.Header.Set("Content-Type", "application/octet-stream")
req.Header.Set("Content-Length", strconv.FormatInt(size, 10))
resp, err := cli.httpClient.Do(req)
if err != nil {
return ouloc, err
}
resp.Body.Close()
if resp.StatusCode != http.StatusAccepted {
err = fmt.Errorf("Upload not accepted, code %d", resp.StatusCode)
return ouloc, err
}
ouloc = resp.Header.Get("Location")
if ouloc == "" {
err := fmt.Errorf("Empty blob location declaration")
return ouloc, err
}
return ouloc, err
}
+1 -1
View File
@@ -12,7 +12,7 @@ import (
func (cli *Client) PutManifest(ctx context.Context, rawrepo, tag string, man []byte, mime string) error {
var err error
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return err
}
+1 -1
View File
@@ -12,7 +12,7 @@ func (cli *Client) PutUpload(ctx context.Context, rawrepo string, src io.Reader,
var err error
var bloc string
ref, err := NewRepository(rawrepo)
ref, err := NewReferer(rawrepo)
if err != nil {
return bloc, err
}
+10 -10
View File
@@ -5,14 +5,14 @@ import (
"strings"
)
type Repository struct {
type Referer struct {
urlobj *url.URL
user, pass string
base string
}
func NewRepository(rawrepo string) (*Repository, error) {
repo := &Repository{}
func NewReferer(rawrepo string) (*Referer, error) {
repo := &Referer{}
if !strings.Contains(rawrepo, "://") {
rawrepo = "https://" + rawrepo
}
@@ -33,27 +33,27 @@ func NewRepository(rawrepo string) (*Repository, error) {
return repo, err
}
func (repo *Repository) String() string {
func (repo *Referer) String() string {
curl := repo.urlobj.JoinPath(repo.base)
return curl.String()
}
func (repo *Repository) Manifest(tag string) string {
func (repo *Referer) Manifest(tag string) string {
curl := repo.urlobj.JoinPath("/v2", repo.base, "/manifests", tag)
return curl.String()
}
func (repo *Repository) Blob(digest string) string {
func (repo *Referer) Blob(digest string) string {
curl := repo.urlobj.JoinPath("/v2", repo.base, "/blobs", digest)
return curl.String()
}
func (repo *Repository) Upload() string {
func (repo *Referer) Upload() string {
curl := repo.urlobj.JoinPath("/v2", repo.base, "/blobs/uploads/")
return curl.String()
}
func (repo *Repository) Patch(loc string) (string, error) {
func (repo *Referer) Patch(loc string) (string, error) {
var curl *url.URL
var out string
var err error
@@ -73,7 +73,7 @@ func (repo *Repository) Patch(loc string) (string, error) {
return out, err
}
func (repo *Repository) Put(loc, digest string) (string, error) {
func (repo *Referer) Put(loc, digest string) (string, error) {
var curl *url.URL
var out string
var err error
@@ -95,6 +95,6 @@ func (repo *Repository) Put(loc, digest string) (string, error) {
return out, err
}
func (repo *Repository) Userinfo() (string, string) {
func (repo *Referer) Userinfo() (string, string) {
return repo.user, repo.pass
}
@@ -10,7 +10,7 @@ import (
)
func xxxTestResrerer(t *testing.T) {
ref, err := NewRepository("registry.example.com/lib/alpine")
ref, err := NewReferer("registry.example.com/lib/alpine")
require.NoError(t, err)
fmt.Printf("Manifest:\t%s\n", ref.Manifest("3.30.0"))
-56
View File
@@ -1,56 +0,0 @@
package rpccli
import (
"net/url"
"path"
"strconv"
"strings"
)
type Referer struct {
urlobj *url.URL
user, pass string
obj, oper string
}
func NewReferer(hostname, object, operation string) (*Referer, error) {
repo := &Referer{
obj: object,
oper: operation,
}
if !strings.Contains(hostname, "://") {
hostname = "https://" + hostname
}
urlobj, err := url.Parse(hostname)
if err != nil {
return repo, err
}
if urlobj.User != nil {
repo.user = urlobj.User.Username()
repo.pass, _ = urlobj.User.Password()
urlobj.User = nil
}
repo.resource = path.Join("/", urlobj.Path)
urlobj.Path = "/"
repo.urlobj = urlobj
return repo, err
}
func (repo *Referer) Raw() string {
return path.Join(repo.urlobj.Host, "/v3/api/", repo.obj, repo.oper)
}
func (repo *Referer) URI(object, operation) string {
curl := repo.urlobj.JoinPath("/v3/api/", object, operation)
return curl.String()
}
func (repo *Referer) Userinfo() (string, string) {
return repo.user, repo.pass
}
func (repo *Referer) SetUserinfo(user, pass string) {
if user != "" && pass != "" {
repo.user, repo.pass = user, pass
}
}