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
|
package eduvpn
import (
"encoding/json"
"errors"
)
type Server struct {
BaseURL string `json:"base_url"`
Endpoints *ServerEndpoints `json:"endpoints"`
OAuth *OAuth `json:"oauth"`
Profiles *ServerProfileInfo `json:"profiles"`
}
type ServerProfile struct {
ID string `json:"profile_id"`
DisplayName string `json:"display_name"`
VPNProtoList []string `json:"vpn_proto_list"`
DefaultGateway bool `json:"default_gateway"`
}
type ServerProfileInfo struct {
Current uint8 `json:"current_profile"`
Info struct {
ProfileList []ServerProfile `json:"profile_list"`
} `json:"info"`
}
type ServerEndpointList struct {
API string `json:"api_endpoint"`
Authorization string `json:"authorization_endpoint"`
Token string `json:"token_endpoint"`
}
// Struct that defines the json format for /.well-known/vpn-user-portal"
type ServerEndpoints struct {
API struct {
V2 ServerEndpointList `json:"http://eduvpn.org/api#2"`
V3 ServerEndpointList `json:"http://eduvpn.org/api#3"`
} `json:"api"`
V string `json:"v"`
}
func (server *Server) Initialize(url string) error {
if !GetVPNState().HasTransition(CHOSEN_SERVER) {
return errors.New("cannot choose a server")
}
server.BaseURL = url
endpointsErr := server.GetEndpoints()
if endpointsErr != nil {
return endpointsErr
}
GetVPNState().GoTransition(CHOSEN_SERVER, "Chosen server")
return nil
}
// FIXME: Check validity of tokens
func (server *Server) IsAuthenticated() bool {
return server.OAuth != nil
// return GetVPNState().HasTransition(SERVER_NOT_AUTHENTICATED)
}
func (server *Server) GetEndpoints() error {
url := server.BaseURL + "/.well-known/vpn-user-portal"
_, body, bodyErr := HTTPGet(url)
if bodyErr != nil {
return bodyErr
}
endpoints := &ServerEndpoints{}
jsonErr := json.Unmarshal(body, &endpoints)
if jsonErr != nil {
return jsonErr
}
server.Endpoints = endpoints
return nil
}
func (profiles *ServerProfileInfo) getCurrentProfile() (*ServerProfile, error) {
if profiles.Info.ProfileList == nil {
return nil, errors.New("No server profiles")
}
if (int)(profiles.Current) >= len(profiles.Info.ProfileList) {
return nil, errors.New("Invalid profile")
}
return &profiles.Info.ProfileList[profiles.Current], nil
}
func (profile *ServerProfile) supportsWireguard() bool {
for _, proto := range profile.VPNProtoList {
if proto == "wireguard" {
return true
}
}
return false
}
func (server *Server) GetCurrentProfile() (*ServerProfile, error) {
if server.Profiles == nil {
return nil, errors.New("No server profiles found")
}
return server.Profiles.getCurrentProfile()
}
func (server *Server) GetConfig() (string, error) {
infoErr := server.APIInfo()
if infoErr != nil {
return "", infoErr
}
profile, profileErr := server.GetCurrentProfile()
if profileErr != nil {
return "", profileErr
}
if profile.supportsWireguard() {
return server.WireguardGetConfig()
}
return server.OpenVPNGetConfig()
}
|