forked from mattn/lastpass-go
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathvault.go
213 lines (182 loc) · 6.37 KB
/
vault.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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
package lastpass
import (
"bytes"
"encoding/xml"
"fmt"
"github.com/pkg/errors"
"io"
"net/url"
"strings"
)
var (
ErrAccountNotFound = fmt.Errorf("account not found")
)
type Vault struct {
sesh *session
email string
opts *ConfigOptions
}
// New logs into LastPass and returns a new Vault
func New(email, password string, opts ...ConfigFunc) (*Vault, error) {
configOpts := new(ConfigOptions)
for _, opt := range opts {
opt(configOpts)
}
session, err := login(email, password, configOpts.multiFactor)
if err != nil {
return nil, err
}
// try to "actually" login because login does
// not check for valid username/password combos
_, err = fetch(session)
if err != nil {
return nil, err
}
return &Vault{sesh: session, email: email, opts: configOpts}, nil
}
// Email returns the email associated with the vault
func (lp Vault) Email() string {
return lp.email
}
// GetAccounts returns all accounts in the LastPass vault
func (lp Vault) GetAccounts() ([]*Account, error) {
blob, err := fetch(lp.sesh)
if err != nil {
return nil, err
}
chunks, err := extractChunks(bytes.NewReader(blob.bytes), []uint32{chunkIdFromString("ACCT")})
if err != nil {
return nil, err
}
accountChunks := chunks[chunkIdFromString("ACCT")]
accs := make([]*Account, len(accountChunks))
for i, chunk := range accountChunks {
account, err := parseAccount(bytes.NewReader(chunk), lp.sesh.key)
if err != nil {
return nil, err
}
accs[i] = account
}
return accs, nil
}
// GetAccount gets LastPass account by unique ID
// If not found, returns ErrAccountNotFound error
func (lp Vault) GetAccount(id string) (*Account, error) {
accs, err := lp.Search(id, Id, CaseInsensitive)
if err != nil {
return nil, err
} else if accs == nil || len(accs) == 0 {
return nil, ErrAccountNotFound
}
return accs[0], nil
}
// Search looks for LastPass accounts matching given args.
func (lp Vault) Search(value string, field Field, method SearchMethod) ([]*Account, error) {
accs, err := lp.GetAccounts()
if err != nil {
return nil, err
}
matchedAccounts := []*Account{}
matchFunc := matchFuncs[method]
for _, acc := range accs {
if matchFunc(getValue(*acc, field), value) {
matchedAccounts = append(matchedAccounts, acc)
}
}
return matchedAccounts, nil
}
// UpdateAccount syncs the LastPass vault all of the fields in the
// account variable.
func (lp *Vault) UpdateAccount(account *Account) (*Account, error) {
_, err := lp.upsertAccount(account)
return account, err
}
// CreateAccount sync LastPass vault with the account info given.
// The return value is the struct with an added Account ID
func (lp *Vault) CreateAccount(account *Account) (*Account, error) {
account.Id = "0"
resp, err := lp.upsertAccount(account)
if err != nil {
return nil, err
}
// https://github.com/gnewton/chidley
var response struct {
Result struct {
AttrAcctname1 string `xml:" acctname1,attr" json:",omitempty"`
AttrAcctname2 string `xml:" acctname2,attr" json:",omitempty"`
AttrAcctname3 string `xml:" acctname3,attr" json:",omitempty"`
AttrAcctname4 string `xml:" acctname4,attr" json:",omitempty"`
AttrAcctname5 string `xml:" acctname5,attr" json:",omitempty"`
AttrAcctname6 string `xml:" acctname6,attr" json:",omitempty"`
AttrAccts_version string `xml:" accts_version,attr" json:",omitempty"`
AttrAction string `xml:" action,attr" json:",omitempty"`
AttrAid string `xml:" aid,attr" json:",omitempty"`
AttrCaptcha_id string `xml:" captcha_id,attr" json:",omitempty"`
AttrCount string `xml:" count,attr" json:",omitempty"`
AttrCustom_js string `xml:" custom_js,attr" json:",omitempty"`
AttrDeleted string `xml:" deleted,attr" json:",omitempty"`
AttrEditlink string `xml:" editlink,attr" json:",omitempty"`
AttrFav string `xml:" fav,attr" json:",omitempty"`
AttrGrouping string `xml:" grouping,attr" json:",omitempty"`
AttrLasttouch string `xml:" lasttouch,attr" json:",omitempty"`
AttrLaunchjs string `xml:" launchjs,attr" json:",omitempty"`
AttrLocalupdate string `xml:" localupdate,attr" json:",omitempty"`
AttrMsg string `xml:" msg,attr" json:",omitempty"`
AttrPwprotect string `xml:" pwprotect,attr" json:",omitempty"`
AttrRemoteshare string `xml:" remoteshare,attr" json:",omitempty"`
AttrSubmit_id string `xml:" submit_id,attr" json:",omitempty"`
AttrUrid string `xml:" urid,attr" json:",omitempty"`
AttrUrl string `xml:" url,attr" json:",omitempty"`
AttrUsername string `xml:" username,attr" json:",omitempty"`
} `xml:" result,omitempty" json:"result,omitempty"`
}
if err = xml.NewDecoder(strings.NewReader(resp)).Decode(&response); err != nil && err != io.EOF {
return nil, err
}
account.Id = response.Result.AttrAid
return account, nil
}
func (lp *Vault) upsertAccount(account *Account) (string, error) {
bUrl := buildLastPassURL("show_website.php")
vals, err := account.encrypt(lp.sesh.key)
if err != nil {
return "", errors.Wrap(err, "failed to encrypt account")
}
return post(bUrl, lp.sesh, vals)
}
// DeleteAccount removes an account from the LastPass vault
// by the Account ID
func (lp *Vault) DeleteAccount(account *Account) error {
bUrl := buildLastPassURL("show_website.php")
values := &url.Values{
"extjs": []string{"1"},
"token": []string{"lp.sesh.token"},
"delete": []string{"1"},
"aid": []string{account.Id},
}
resp, err := post(bUrl, lp.sesh, values)
if err != nil {
return err
}
var response struct {
Result struct {
AttrAccts_version string `xml:" accts_version,attr" json:",omitempty"`
AttrAction string `xml:" action,attr" json:",omitempty"`
AttrAid string `xml:" aid,attr" json:",omitempty"`
AttrLocalupdate string `xml:" localupdate,attr" json:",omitempty"`
AttrMsg string `xml:" msg,attr" json:",omitempty"`
} `xml:" result,omitempty" json:"result,omitempty"`
}
if err = xml.NewDecoder(strings.NewReader(resp)).Decode(&response); err != nil && err != io.EOF {
return err
}
if response.Result.AttrMsg != "accountdeleted" {
return fmt.Errorf("failed to delete account %s: %s", account.Id, response.Result.AttrMsg)
}
return nil
}
// DeleteAccountById removes an account from LastPass
// by the Account ID
func (lp *Vault) DeleteAccountById(id string) error {
return lp.DeleteAccount(&Account{Id: id})
}