summaryrefslogtreecommitdiff
path: root/src/server.go
blob: d5120490710bfc4dfe3934ded6ee5cab97eb8626 (plain)
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
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 {
	server.BaseURL = url
	endpointsErr := server.GetEndpoints()
	if endpointsErr != nil {
		return endpointsErr
	}
	return nil
}

// FIXME: Check validity of tokens
func (server *Server) IsAuthenticated() bool {
	return server.OAuth != nil
}

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()
}