This repository has been archived by the owner on Oct 30, 2023. It is now read-only.
/
client.go
124 lines (112 loc) · 3.61 KB
/
client.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
// client.go - Katzenpost client registration library
// Copyright (C) 2018 David Stainton.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
// Package client provides a library for registering Katzenpost
// clients with a specific mixnet Provider.
package client
import (
"fmt"
"net/http"
"net/url"
"github.com/katzenpost/core/crypto/ecdh"
"github.com/katzenpost/server/registration"
"golang.org/x/net/proxy"
)
// Options are optional parameters to configure
// the registration client. Default values are used
// when a nil Options pointer is passed to New.
type Options struct {
// Scheme selects the HTTP scheme
// which is either HTTP or HTTPS
Scheme string
// UseSocks is set to true if the specified
// SOCKS proxy is to be used for dialing.
UseSocks bool
// SocksNetwork is the network that the
// optional SOCKS port is listening on
// which is usually "unix" or "tcp".
SocksNetwork string
// SocksAddress is the address of the SOCKS port.
SocksAddress string
}
var defaultOptions = Options{
Scheme: "https",
}
// Client handles mixnet Provider account registration.
type Client struct {
url *url.URL
options *Options
client *http.Client
}
// New creates a new Client with the provided configuration.
func New(address string, options *Options) (*Client, error) {
if options == nil {
options = &defaultOptions
}
client := new(http.Client)
if options.UseSocks {
dialer, err := proxy.SOCKS5(options.SocksNetwork, options.SocksAddress, nil, proxy.Direct)
if err != nil {
return nil, err
}
tr := &http.Transport{
Dial: dialer.Dial,
}
client = &http.Client{Transport: tr}
}
c := &Client{
url: &url.URL{
Scheme: options.Scheme,
Host: address,
Path: registration.URLBase,
},
options: options,
client: client,
}
return c, nil
}
func (c *Client) RegisterAccountWithIdentityAndLinkKey(user string, linkKey *ecdh.PublicKey, identityKey *ecdh.PublicKey) error {
formData := url.Values{
registration.VersionField: {registration.Version},
registration.CommandField: {registration.RegisterLinkAndIdentityCommand},
registration.UserField: {user},
registration.LinkKeyField: {linkKey.String()},
registration.IdentityKeyField: {identityKey.String()},
}
response, err := c.client.PostForm(c.url.String(), formData)
if err != nil {
return err
}
if response.StatusCode != http.StatusOK {
return fmt.Errorf("Registration failure: received status code %d", response.StatusCode)
}
return nil
}
func (c *Client) RegisterAccountWithLinkKey(user string, linkKey *ecdh.PublicKey) error {
formData := url.Values{
registration.VersionField: {registration.Version},
registration.CommandField: {registration.RegisterLinkCommand},
registration.UserField: {user},
registration.LinkKeyField: {linkKey.String()},
}
response, err := c.client.PostForm(c.url.String(), formData)
if err != nil {
return err
}
if response.StatusCode != http.StatusOK {
return fmt.Errorf("Registration failure: received status code %d", response.StatusCode)
}
return nil
}