-
Notifications
You must be signed in to change notification settings - Fork 6
/
sequel_pro_flow.go
122 lines (101 loc) · 3.93 KB
/
sequel_pro_flow.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
package amanar
import (
"io/ioutil"
"fmt"
"log"
"errors"
"howett.net/plist"
)
const SEQUEL_PRO_PLIST_FORMAT = plist.XMLFormat
type SequelProRootPlist struct {
FavoritesRoot struct {
IsExpanded bool `plist:"IsExpanded"`
Name string `plist:"Name"`
Children []SequelProPlistItem `plist:"Children"`
} `plist:"Favorites Root"`
}
type SequelProPlistItem struct {
ColorIndex int64 `plist:"colorIndex"`
Database string `plist:"database"`
Host string `plist:"host"`
Id int64 `plist:"id"`
Name string `plist:"name"`
Port string `plist:"port"`
Socket string `plist:"socket"`
SSHHost string `plist:"sshHost"`
SSHKeyLocation string `plist:"sshKeyLocation"`
SSHKeyLocationEnabled int `plist:"sshKeyLocationEnabled"`
SSHPort string `plist:"sshPort"`
SSHUser string `plist:"sshUser"`
SSLCACertFileLocation string `plist:"sslcaCertFileLocation"`
SSLCACertFileLocationEnabled int `plist:"sslcaCertFileLocationEnabled"`
SSLCertificateFileLocation string `plist:"sslCertificateFileLocation"`
SSLCertificateFileLocationEnabled int `plist:"sslCertificateFileLocationEnabled"`
SSLKeyFileLocation string `plist:"sslKeyFileLocation"`
SSLKeyFileLocationEnabled int `plist:"sslKeyFileLocationEnabled"`
Type int `plist:"type"`
UseSSL int `plist:"useSSL"`
User string `plist:"user"`
}
func NewSequelProFlow(config *SequelProDatasource) (spf *SequelProFlow, err error) {
bytes, err := ioutil.ReadFile(config.SequelProPlistPath)
if err != nil {
return
}
sequelPlist := SequelProRootPlist{}
_, err = plist.Unmarshal(bytes, &sequelPlist)
if err != nil {
return
}
return &SequelProFlow{
SequelProDatasource: *config,
plist: sequelPlist,
}, nil
}
type SequelProFlow struct {
SequelProDatasource
plist SequelProRootPlist
credentials *Credentials
}
func (sp *SequelProFlow) findPlistItem() (plistItem *SequelProPlistItem, foundItem bool) {
children := sp.plist.FavoritesRoot.Children
// N.B. the iteratees of range will be copied values. This allows
// us to refer to the indexed children as actual pointers
for i := 0; i < len(children); i++ {
if fmt.Sprintf("%d", children[i].Id) == sp.DatabaseUUID {
plistItem = &children[i]
foundItem = true
break
}
}
return
}
func (sp *SequelProFlow) Name() string {
return "SEQUEL PRO"
}
func (sp *SequelProFlow) UpdateWithCredentials(credentials *Credentials) (err error) {
plistItem, found := sp.findPlistItem()
if !found {
return errors.New(fmt.Sprintf("[SEQUEL PRO] Could not find plist item for database UUID %s", sp.DatabaseUUID))
}
plistItem.User = credentials.Username
sp.credentials = credentials
return nil
}
func (sp *SequelProFlow) PersistChanges() (err error) {
plistItem, found := sp.findPlistItem()
if !found {
return errors.New("Could not find a matching Sequel Database for that Sequel UUID.")
}
// These two values need to be synchronized for Sequel Pro to be able to
// read the correct keychain value.
service := fmt.Sprintf("Sequel Pro : %s (%d)", plistItem.Name, plistItem.Id)
account := fmt.Sprintf("%s@%s/%s", plistItem.User, plistItem.Host, plistItem.Database)
log.Printf("[SEQUEL PRO] Persisting username %s and password %s to service %s and account %s", plistItem.User, sp.credentials.Password, service, account)
bytes, err := plist.Marshal(sp.plist, SEQUEL_PRO_PLIST_FORMAT)
if err != nil {
return
}
ioutil.WriteFile(sp.SequelProPlistPath, bytes, 0644)
return CreateOrUpdateKeychainEntriesForService(service, account, sp.credentials.Password, []string{})
}