-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathaddressbook.go
132 lines (116 loc) · 3.1 KB
/
addressbook.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
package o3
import (
"bytes"
"encoding/csv"
"encoding/hex"
"fmt"
)
import "os"
// ThreemaContact is the core contact type, comprising of
// an ID, a long-term public key, and an optional Name
type ThreemaContact struct {
ID [8]byte
Name string
LPK [32]byte
}
func (tc ThreemaContact) String() string {
return string(tc.ID[:])
}
// AddressBook is the register of ThreemaContacts
type AddressBook struct {
contacts map[string]ThreemaContact
}
func (a AddressBook) slice() [][]string {
buf := make([][]string, len(a.contacts))
i := 0
for _, contact := range a.contacts {
buf[i] = make([]string, 3)
buf[i][0] = string(contact.ID[:])
buf[i][1] = contact.Name
buf[i][2] = hex.EncodeToString(contact.LPK[:])
i++
}
return buf
}
func (a *AddressBook) initializeMap(c int) {
a.contacts = make(map[string]ThreemaContact, c)
}
// Import takes a two-dimensional slice of strings and imports it
// field by field into the address book.
// Fields have to be in the order "ID, Name, LPK" or the function will
// return an error
func (a *AddressBook) Import(contacts [][]string) error {
a.contacts = make(map[string]ThreemaContact, len(contacts))
if a.contacts == nil {
a.initializeMap(len(contacts))
}
for l, c := range contacts {
// log.Printf("%#v\n", c)
id := c[0]
contact := ThreemaContact{Name: c[1]}
n := copy(contact.ID[:], id[:])
if n != 8 {
return fmt.Errorf("line %d: invalid ID length: %d", l, n)
}
lpk, err := hex.DecodeString(c[2])
if err != nil {
return err
}
n = copy(contact.LPK[0:32], lpk[0:32])
if n != 32 {
return fmt.Errorf("line %d: invalid pubKey length: %d", l, n)
}
a.contacts[id] = contact
}
return nil
}
// ImportFrom imports an address book stored in a CSV file
func (a *AddressBook) ImportFrom(filename string) error {
file, err := os.Open(filename)
if err != nil {
return err
}
defer file.Close()
rdr := csv.NewReader(file)
lines, err := rdr.ReadAll()
// log.Printf("Read lines: %#v\n", lines)
if err != nil {
return err
}
return a.Import(lines)
}
// SaveTo stores the AddressBook in the file with the given name
// in CSV format
func (a AddressBook) SaveTo(filename string) error {
file, err := os.Create(filename)
if err != nil {
return err
}
defer file.Close()
wrtr := csv.NewWriter(file)
slice := a.slice()
return wrtr.WriteAll(slice)
}
// Add takes a ThreemaContact and adds it to the AddressBook
func (a *AddressBook) Add(c ThreemaContact) {
id := string(c.ID[:])
if a.contacts == nil {
a.initializeMap(1)
}
a.contacts[id] = c
}
// Get returns a ThreemaContact to a given ID. It returns an empty ThreemaContact
// if no entry is found. The second parameter can be used to check if
// retrieval was successful
func (a AddressBook) Get(id string) (ThreemaContact, bool) {
contact := a.contacts[id]
//checking if an empty ThreemaContact was returned
if bytes.Equal(contact.ID[:], []byte{0, 0, 0, 0, 0, 0, 0, 0}) {
return contact, false
}
return contact, true
}
// Contacts returns the map of id strings to contact structs of all contacts in the address book
func (a AddressBook) Contacts() map[string]ThreemaContact {
return a.contacts
}