Skip to content

Commit

Permalink
All: Refactor to Discovery v3
Browse files Browse the repository at this point in the history
And rename `instance` to `provider`
  • Loading branch information
jwijenbergh committed Dec 4, 2024
1 parent ed1a43c commit 8645bad
Show file tree
Hide file tree
Showing 10 changed files with 557 additions and 330 deletions.
36 changes: 18 additions & 18 deletions cmd/geteduroam-cli/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,10 +17,10 @@ import (

"github.com/geteduroam/linux-app/internal/discovery"
"github.com/geteduroam/linux-app/internal/handler"
"github.com/geteduroam/linux-app/internal/instance"
"github.com/geteduroam/linux-app/internal/log"
"github.com/geteduroam/linux-app/internal/network"
"github.com/geteduroam/linux-app/internal/notification"
"github.com/geteduroam/linux-app/internal/provider"
"github.com/geteduroam/linux-app/internal/utils"
"github.com/geteduroam/linux-app/internal/version"
)
Expand Down Expand Up @@ -73,8 +73,8 @@ func ask(prompt string, validator func(input string) bool) string {
}
}

// filteredOrganizations gets the instances as filtered by the user
func filteredOrganizations(orgs *instance.Instances, q string) (f *instance.Instances) {
// filteredOrganizations gets the providers as filtered by the user
func filteredOrganizations(orgs *provider.Providers, q string) (f *provider.Providers) {
for {
empties := 0
x := ask(q, func(x string) bool {
Expand Down Expand Up @@ -116,8 +116,8 @@ func validateRange(input string, n int) bool {
return true
}

// organization gets an organization/instance from the user
func organization(orgs *instance.Instances) *instance.Instance {
// organization gets an organization/provider from the user
func organization(orgs *provider.Providers) *provider.Provider {
_, h, err := term.GetSize(0)
if err != nil {
slog.Warn("Could not get height")
Expand All @@ -128,7 +128,7 @@ func organization(orgs *instance.Instances) *instance.Instance {
for {
if len(*f) > h-3 {
for _, c := range *f {
fmt.Printf("%s\n", c.Name)
fmt.Printf("%s\n", c.Name.Get())
}
fmt.Println("\nList is long...")
f = filteredOrganizations(f, "Please refine your search: ")
Expand All @@ -138,7 +138,7 @@ func organization(orgs *instance.Instances) *instance.Instance {
}
fmt.Println("\nFound the following matches: ")
for n, c := range *f {
fmt.Printf("[%d] %s\n", n+1, c.Name)
fmt.Printf("[%d] %s\n", n+1, c.Name.Get())
}
input := ask("\nPlease enter a choice for the organisation: ", func(input string) bool {
return validateRange(input, len(*f))
Expand All @@ -152,15 +152,15 @@ func organization(orgs *instance.Instances) *instance.Instance {
}

// profile gets a profile for a list of profiles by asking the user one if there are multiple
func profile(profiles []instance.Profile) *instance.Profile {
func profile(profiles []provider.Profile) *provider.Profile {
// Only one profile, return it immediately
if len(profiles) == 1 {
return &profiles[0]
}
// Multiple profiles found, we need to get the right one
fmt.Println("Found the following profiles: ")
for n, c := range profiles {
fmt.Printf("[%d] %s\n", n+1, c.Name)
fmt.Printf("[%d] %s\n", n+1, c.Name.Get())
}
input := ask("Please enter a choice for the profile: ", func(input string) bool {
return validateRange(input, len(profiles))
Expand Down Expand Up @@ -312,7 +312,7 @@ func file(metadata []byte) (*time.Time, error) {
}

// direct does the handling for the direct flow
func direct(p *instance.Profile) {
func direct(p *provider.Profile) {
config, err := p.EAPDirect()
if err != nil {
slog.Error("Could not obtain eap config", "error", err)
Expand All @@ -330,7 +330,7 @@ func direct(p *instance.Profile) {
}

// redirect does the handling for the redirect flow
func redirect(p *instance.Profile) {
func redirect(p *provider.Profile) {
r, err := p.RedirectURI()
if err != nil {
slog.Error("Failed to complete the flow, no redirect URI is available")
Expand All @@ -347,7 +347,7 @@ func redirect(p *instance.Profile) {
}

// oauth does the handling for the OAuth flow
func oauth(p *instance.Profile) *time.Time {
func oauth(p *provider.Profile) *time.Time {
config, err := p.EAPOAuth(context.Background(), func(url string) {
fmt.Println("Your browser has been opened to authorize the client")
fmt.Println("Or copy and paste the following url:", url)
Expand Down Expand Up @@ -384,10 +384,10 @@ func doLocal(filename string) *time.Time {

func doDiscovery() *time.Time {
c := discovery.NewCache()
i, err := c.Instances()
i, err := c.Providers()
if err != nil {
slog.Error("Failed to get instances from discovery", "error", err)
fmt.Printf("Failed to get instances from discovery %v\n", err)
slog.Error("Failed to get providers from discovery", "error", err)
fmt.Printf("Failed to get providers from discovery %v\n", err)
os.Exit(1)
}

Expand All @@ -397,11 +397,11 @@ func doDiscovery() *time.Time {
// TODO: This switch statement should probably be moved to the profile code
// By providing an "EAP" method on profile
switch p.Flow() {
case instance.DirectFlow:
case provider.DirectFlow:
direct(p)
case instance.RedirectFlow:
case provider.RedirectFlow:
redirect(p)
case instance.OAuthFlow:
case provider.OAuthFlow:
return oauth(p)
}
return nil
Expand Down
4 changes: 4 additions & 0 deletions cmd/geteduroam-gui/list.go
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,10 @@ func (s *SelectList) Add(idx int, label string) {
strobj.SetData("model-index", uintptr(idx))
}

func (s *SelectList) Remove(idx int) {
s.store.Remove(uint(idx))
}

func (s *SelectList) Show() {
s.win.Show()
}
Expand Down
70 changes: 49 additions & 21 deletions cmd/geteduroam-gui/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,34 +21,55 @@ import (

"github.com/geteduroam/linux-app/internal/discovery"
"github.com/geteduroam/linux-app/internal/handler"
"github.com/geteduroam/linux-app/internal/instance"
"github.com/geteduroam/linux-app/internal/log"
"github.com/geteduroam/linux-app/internal/network"
"github.com/geteduroam/linux-app/internal/provider"
"github.com/geteduroam/linux-app/internal/version"
)

type serverList struct {
sync.Mutex
store *gtk.StringList
instances instance.Instances
providers provider.Providers
list *SelectList
custom bool
}

func (s *serverList) get(idx int) (*instance.Instance, error) {
if idx < 0 || idx > len(s.instances) {
func (s *serverList) get(idx int, query string) (*provider.Provider, error) {
if s.custom && idx == len(s.providers) {
// TODO: add context
return provider.Custom(context.Background(), query)
}
if idx < 0 || idx > len(s.providers) {
return nil, errors.New("index out of range")
}
return &s.instances[idx], nil
return &s.providers[idx], nil
}

func (s *serverList) Fill() {
s.Lock()
defer s.Unlock()
for idx, inst := range s.instances {
s.list.Add(idx, inst.Name)
for idx, inst := range s.providers {
s.list.Add(idx, inst.Name.Get())
}
}

func (s *serverList) AddCustom(label string) {
if s.custom {
s.RemoveCustom()
}
s.list.Add(len(s.providers), label)
s.custom = true
}

func (s *serverList) RemoveCustom() {
if !s.custom {
return
}
s.list.Remove(len(s.providers))
s.custom = false
}

type mainState struct {
app *adw.Application
builder *gtk.Builder
Expand Down Expand Up @@ -93,7 +114,7 @@ func (m *mainState) file(metadata []byte) (*time.Time, error) {
return h.Configure(metadata)
}

func (m *mainState) direct(p instance.Profile) error {
func (m *mainState) direct(p provider.Profile) error {
config, err := p.EAPDirect()
if err != nil {
return err
Expand All @@ -114,7 +135,7 @@ func (m *mainState) local(path string) (*time.Time, error) {
return v, nil
}

func (m *mainState) oauth(ctx context.Context, p instance.Profile) (*time.Time, error) {
func (m *mainState) oauth(ctx context.Context, p provider.Profile) (*time.Time, error) {
ctx, cancel := context.WithCancel(ctx)
defer cancel()
config, err := p.EAPOAuth(ctx, func(url string) {
Expand All @@ -135,31 +156,31 @@ func (m *mainState) oauth(ctx context.Context, p instance.Profile) (*time.Time,
return m.file(config)
}

func (m *mainState) rowActivated(sel instance.Instance) {
func (m *mainState) rowActivated(sel provider.Provider) {
var page gtk.Box
m.builder.GetObject("searchPage").Cast(&page)
defer page.Unref()
l := NewLoadingPage(m.builder, m.stack, "Loading organization details...", nil)
l.Initialize()
ctx := context.Background()
chosen := func(p instance.Profile) (err error) {
chosen := func(p provider.Profile) (err error) {
defer func() {
err = ensureContextError(ctx, err)
}()
var valid *time.Time
var isredirect bool
switch p.Flow() {
case instance.DirectFlow:
case provider.DirectFlow:
err = m.direct(p)
if err != nil {
return err
}
case instance.OAuthFlow:
case provider.OAuthFlow:
valid, err = m.oauth(ctx, p)
if err != nil {
return err
}
case instance.RedirectFlow:
case provider.RedirectFlow:
isredirect = true
url, err := p.RedirectURI()
if err != nil {
Expand All @@ -177,7 +198,7 @@ func (m *mainState) rowActivated(sel instance.Instance) {
})
return nil
}
cb := func(p instance.Profile) {
cb := func(p provider.Profile) {
err := chosen(p)
if err != nil {
l.Hide()
Expand All @@ -200,19 +221,19 @@ func (m *mainState) initList() {
defer list.Unref()

cache := discovery.NewCache()
inst, err := cache.Instances()
inst, err := cache.Providers()
if err != nil {
m.ShowError(err)
return
}
m.servers.instances = *inst
m.servers.providers = *inst

var search gtk.SearchEntry
m.builder.GetObject("searchBox").Cast(&search)
defer search.Unref()

activated := func(idx int) {
inst, err := m.servers.get(idx)
inst, err := m.servers.get(idx, search.GetText())
// TODO: handle error
if err != nil {
m.ShowError(err)
Expand All @@ -222,11 +243,11 @@ func (m *mainState) initList() {
}

sorter := func(a, b string) int {
return instance.SortNames(a, b, search.GetText())
return provider.SortNames(a, b, search.GetText())
}

m.servers.list = NewSelectList(m.scroll, &list, activated, sorter).WithFiltering(func(a string) bool {
return instance.FilterSingle(a, search.GetText())
return provider.FilterSingle(a, search.GetText())
})

// Fill the servers in the select list
Expand All @@ -238,10 +259,17 @@ func (m *mainState) initList() {
changedcb := func(_ gtk.SearchEntry) {
// TODO len returns length in bytes
// utf8.RuneCountInString() counts number of characters (runes)
if len(search.GetText()) <= 2 {
query := search.GetText()
if len(query) <= 2 {
m.servers.list.Hide()
return
}
// url entered
if strings.Count(query, ".") >= 2 {
m.servers.AddCustom(search.GetText())
} else {
m.servers.RemoveCustom()
}
m.servers.list.Changed()
m.servers.list.Show()
}
Expand Down
12 changes: 6 additions & 6 deletions cmd/geteduroam-gui/profile.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,20 +3,20 @@ package main
import (
"golang.org/x/exp/slog"

"github.com/geteduroam/linux-app/internal/instance"
"github.com/geteduroam/linux-app/internal/provider"
"github.com/jwijenbergh/puregotk/v4/adw"
"github.com/jwijenbergh/puregotk/v4/gtk"
)

type ProfileState struct {
builder *gtk.Builder
stack *adw.ViewStack
profiles []instance.Profile
success func(instance.Profile)
profiles []provider.Profile
success func(provider.Profile)
sl *SelectList
}

func NewProfileState(builder *gtk.Builder, stack *adw.ViewStack, profiles []instance.Profile, success func(instance.Profile)) *ProfileState {
func NewProfileState(builder *gtk.Builder, stack *adw.ViewStack, profiles []provider.Profile, success func(provider.Profile)) *ProfileState {
return &ProfileState{
builder: builder,
stack: stack,
Expand Down Expand Up @@ -55,7 +55,7 @@ func (p *ProfileState) Initialize() {

sorter := func(a, b string) int {
// Here we have no search query
return instance.SortNames(a, b, "")
return provider.SortNames(a, b, "")
}
activated := func(idx int) {
go p.success(p.profiles[idx])
Expand All @@ -65,7 +65,7 @@ func (p *ProfileState) Initialize() {
p.sl = NewSelectList(&scroll, &list, activated, sorter)

for idx, prof := range p.profiles {
p.sl.Add(idx, prof.Name)
p.sl.Add(idx, prof.Name.Get())
}

p.sl.Setup()
Expand Down
Loading

0 comments on commit 8645bad

Please sign in to comment.