moby--moby/auth/auth.go

195 lines
5.3 KiB
Go
Raw Normal View History

package auth
import (
"encoding/base64"
"encoding/json"
"errors"
"fmt"
"io/ioutil"
"net/http"
"os"
"path"
"strings"
)
// Where we store the config file
const CONFIGFILE = ".dockercfg"
// Only used for user auth + account creation
const INDEXSERVER = "https://index.docker.io/v1/"
2013-06-04 18:00:22 +00:00
//const INDEXSERVER = "http://indexstaging-docker.dotcloud.com/"
var (
2013-06-04 13:51:12 +00:00
ErrConfigFileMissing = errors.New("The Auth config file is missing")
)
type AuthConfig struct {
Username string `json:"username,omitempty"`
Password string `json:"password,omitempty"`
Auth string `json:"auth"`
Email string `json:"email"`
}
type ConfigFile struct {
Configs map[string]AuthConfig `json:"configs,omitempty"`
rootPath string
2013-03-22 12:52:13 +00:00
}
func IndexServerAddress() string {
2013-06-04 18:00:22 +00:00
return INDEXSERVER
}
// create a base64 encoded auth string to store in config
2013-05-30 15:39:43 +00:00
func encodeAuth(authConfig *AuthConfig) string {
authStr := authConfig.Username + ":" + authConfig.Password
msg := []byte(authStr)
encoded := make([]byte, base64.StdEncoding.EncodedLen(len(msg)))
base64.StdEncoding.Encode(encoded, msg)
return string(encoded)
}
// decode the auth string
func decodeAuth(authStr string) (string, string, error) {
decLen := base64.StdEncoding.DecodedLen(len(authStr))
decoded := make([]byte, decLen)
authByte := []byte(authStr)
n, err := base64.StdEncoding.Decode(decoded, authByte)
if err != nil {
return "", "", err
}
if n > decLen {
return "", "", fmt.Errorf("Something went wrong decoding auth config")
}
arr := strings.Split(string(decoded), ":")
if len(arr) != 2 {
return "", "", fmt.Errorf("Invalid auth configuration file")
}
password := strings.Trim(arr[1], "\x00")
return arr[0], password, nil
}
// load up the auth config information and return values
// FIXME: use the internal golang config parser
func LoadConfig(rootPath string) (*ConfigFile, error) {
configFile := ConfigFile{Configs: make(map[string]AuthConfig), rootPath: rootPath}
confFile := path.Join(rootPath, CONFIGFILE)
if _, err := os.Stat(confFile); err != nil {
return &configFile, ErrConfigFileMissing
}
b, err := ioutil.ReadFile(confFile)
if err != nil {
return nil, err
}
if err := json.Unmarshal(b, &configFile.Configs); err != nil {
arr := strings.Split(string(b), "\n")
if len(arr) < 2 {
return nil, fmt.Errorf("The Auth config file is empty")
}
authConfig := AuthConfig{}
origAuth := strings.Split(arr[0], " = ")
authConfig.Username, authConfig.Password, err = decodeAuth(origAuth[1])
if err != nil {
return nil, err
}
origEmail := strings.Split(arr[1], " = ")
authConfig.Email = origEmail[1]
configFile.Configs[IndexServerAddress()] = authConfig
} else {
for k, authConfig := range configFile.Configs {
authConfig.Username, authConfig.Password, err = decodeAuth(authConfig.Auth)
if err != nil {
return nil, err
}
configFile.Configs[k] = authConfig
}
}
return &configFile, nil
}
// save the auth config
func SaveConfig(configFile *ConfigFile) error {
confFile := path.Join(configFile.rootPath, CONFIGFILE)
if len(configFile.Configs) == 0 {
os.Remove(confFile)
return nil
}
for k, authConfig := range configFile.Configs {
authConfig.Auth = encodeAuth(&authConfig)
authConfig.Username = ""
authConfig.Password = ""
configFile.Configs[k] = authConfig
}
b, err := json.Marshal(configFile.Configs)
if err != nil {
return err
}
err = ioutil.WriteFile(confFile, b, 0600)
if err != nil {
return err
}
return nil
}
// try to register/login to the registry server
func Login(authConfig *AuthConfig) (string, error) {
2013-04-24 19:15:34 +00:00
client := &http.Client{}
reqStatusCode := 0
var status string
var reqBody []byte
2013-03-15 21:41:55 +00:00
jsonBody, err := json.Marshal(authConfig)
if err != nil {
2013-04-24 16:11:29 +00:00
return "", fmt.Errorf("Config Error: %s", err)
2013-03-15 21:41:55 +00:00
}
2013-04-02 10:00:21 +00:00
// using `bytes.NewReader(jsonBody)` here causes the server to respond with a 411 status.
b := strings.NewReader(string(jsonBody))
req1, err := http.Post(IndexServerAddress()+"users/", "application/json; charset=utf-8", b)
2013-03-15 01:43:02 +00:00
if err != nil {
2013-04-24 16:11:29 +00:00
return "", fmt.Errorf("Server Error: %s", err)
}
2013-03-15 01:43:02 +00:00
reqStatusCode = req1.StatusCode
2013-03-15 21:41:55 +00:00
defer req1.Body.Close()
reqBody, err = ioutil.ReadAll(req1.Body)
if err != nil {
2013-04-24 16:11:29 +00:00
return "", fmt.Errorf("Server Error: [%#v] %s", reqStatusCode, err)
2013-03-15 21:41:55 +00:00
}
2013-03-15 01:43:02 +00:00
if reqStatusCode == 201 {
2013-05-02 22:39:44 +00:00
status = "Account created. Please use the confirmation link we sent" +
2013-06-14 13:38:51 +00:00
" to your e-mail to activate it."
2013-05-01 20:41:58 +00:00
} else if reqStatusCode == 403 {
2013-05-02 22:39:44 +00:00
return "", fmt.Errorf("Login: Your account hasn't been activated. " +
2013-05-01 20:41:58 +00:00
"Please check your e-mail for a confirmation link.")
} else if reqStatusCode == 400 {
if string(reqBody) == "\"Username or email already exists\"" {
req, err := http.NewRequest("GET", IndexServerAddress()+"users/", nil)
req.SetBasicAuth(authConfig.Username, authConfig.Password)
resp, err := client.Do(req)
if err != nil {
return "", err
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
return "", err
}
if resp.StatusCode == 200 {
2013-06-14 13:38:51 +00:00
status = "Login Succeeded"
} else if resp.StatusCode == 401 {
return "", fmt.Errorf("Wrong login/password, please try again")
} else {
return "", fmt.Errorf("Login: %s (Code: %d; Headers: %s)", body,
resp.StatusCode, resp.Header)
}
} else {
2013-04-24 16:11:29 +00:00
return "", fmt.Errorf("Registration: %s", reqBody)
}
2013-03-15 01:43:02 +00:00
} else {
2013-04-24 16:11:29 +00:00
return "", fmt.Errorf("Unexpected status code [%d] : %s", reqStatusCode, reqBody)
}
return status, nil
}