package http import ( "bytes" "context" "encoding/json" "fmt" "io" "net" "net/http" "net/http/httptest" "testing" "time" "github.com/gorilla/mux" "golang.org/x/exp/maps" "github.com/netbirdio/netbird/management/server" nbgroup "github.com/netbirdio/netbird/management/server/group" "github.com/netbirdio/netbird/management/server/http/api" "github.com/netbirdio/netbird/management/server/jwtclaims" nbpeer "github.com/netbirdio/netbird/management/server/peer" "github.com/stretchr/testify/assert" "github.com/netbirdio/netbird/management/server/mock_server" ) type ctxKey string const ( testPeerID = "test_peer" noUpdateChannelTestPeerID = "no-update-channel" adminUser = "admin_user" regularUser = "regular_user" serviceUser = "service_user" userIDKey ctxKey = "user_id" ) func initTestMetaData(peers ...*nbpeer.Peer) *PeersHandler { peersMap := make(map[string]*nbpeer.Peer) for _, peer := range peers { peersMap[peer.ID] = peer.Copy() } policy := &server.Policy{ ID: "policy", AccountID: "test_id", Name: "policy", Enabled: true, Rules: []*server.PolicyRule{ { ID: "rule", Name: "rule", Enabled: true, Action: "accept", Destinations: []string{"group1"}, Sources: []string{"group1"}, Bidirectional: true, Protocol: "all", Ports: []string{"80"}, }, }, } srvUser := server.NewRegularUser(serviceUser) srvUser.IsServiceUser = true account := &server.Account{ Id: "test_id", Domain: "hotmail.com", Peers: peersMap, Users: map[string]*server.User{ adminUser: server.NewAdminUser(adminUser), regularUser: server.NewRegularUser(regularUser), serviceUser: srvUser, }, Groups: map[string]*nbgroup.Group{ "group1": { ID: "group1", AccountID: "test_id", Name: "group1", Issued: "api", Peers: maps.Keys(peersMap), }, }, Settings: &server.Settings{ PeerLoginExpirationEnabled: true, PeerLoginExpiration: time.Hour, }, Policies: []*server.Policy{policy}, Network: &server.Network{ Identifier: "ciclqisab2ss43jdn8q0", Net: net.IPNet{ IP: net.ParseIP("100.67.0.0"), Mask: net.IPv4Mask(255, 255, 0, 0), }, Serial: 51, }, } return &PeersHandler{ accountManager: &mock_server.MockAccountManager{ UpdatePeerFunc: func(_ context.Context, accountID, userID string, update *nbpeer.Peer) (*nbpeer.Peer, error) { var p *nbpeer.Peer for _, peer := range peers { if update.ID == peer.ID { p = peer.Copy() break } } p.SSHEnabled = update.SSHEnabled p.LoginExpirationEnabled = update.LoginExpirationEnabled p.Name = update.Name return p, nil }, GetPeerFunc: func(_ context.Context, accountID, peerID, userID string) (*nbpeer.Peer, error) { var p *nbpeer.Peer for _, peer := range peers { if peerID == peer.ID { p = peer.Copy() break } } return p, nil }, GetPeersFunc: func(_ context.Context, accountID, userID string) ([]*nbpeer.Peer, error) { return peers, nil }, GetPeerGroupsFunc: func(ctx context.Context, accountID, peerID string) ([]*nbgroup.Group, error) { peersID := make([]string, len(peers)) for _, peer := range peers { peersID = append(peersID, peer.ID) } return []*nbgroup.Group{ { ID: "group1", AccountID: accountID, Name: "group1", Issued: "api", Peers: peersID, }, }, nil }, GetDNSDomainFunc: func() string { return "netbird.selfhosted" }, GetAccountIDFromTokenFunc: func(_ context.Context, claims jwtclaims.AuthorizationClaims) (string, string, error) { return claims.AccountId, claims.UserId, nil }, GetAccountFunc: func(ctx context.Context, accountID string) (*server.Account, error) { return account, nil }, GetAccountByIDFunc: func(ctx context.Context, accountID string, userID string) (*server.Account, error) { return account, nil }, HasConnectedChannelFunc: func(peerID string) bool { statuses := make(map[string]struct{}) for _, peer := range peers { if peer.ID == noUpdateChannelTestPeerID { break } statuses[peer.ID] = struct{}{} } _, ok := statuses[peerID] return ok }, }, claimsExtractor: jwtclaims.NewClaimsExtractor( jwtclaims.WithFromRequestContext(func(r *http.Request) jwtclaims.AuthorizationClaims { userID := r.Context().Value(userIDKey).(string) return jwtclaims.AuthorizationClaims{ UserId: userID, Domain: "hotmail.com", AccountId: "test_id", } }), ), } } // Tests the GetAllPeers endpoint reachable in the route /api/peers // Use the metadata generated by initTestMetaData() to check for values func TestGetPeers(t *testing.T) { peer := &nbpeer.Peer{ ID: testPeerID, Key: "key", IP: net.ParseIP("100.64.0.1"), Status: &nbpeer.PeerStatus{Connected: true}, Name: "PeerName", LoginExpirationEnabled: false, Meta: nbpeer.PeerSystemMeta{ Hostname: "hostname", GoOS: "GoOS", Kernel: "kernel", Core: "core", Platform: "platform", OS: "OS", WtVersion: "development", SystemSerialNumber: "C02XJ0J0JGH7", }, } peer1 := peer.Copy() peer1.ID = noUpdateChannelTestPeerID expectedUpdatedPeer := peer.Copy() expectedUpdatedPeer.LoginExpirationEnabled = true expectedUpdatedPeer.SSHEnabled = true expectedUpdatedPeer.Name = "New Name" expectedPeer1 := peer1.Copy() expectedPeer1.Status.Connected = false tt := []struct { name string expectedStatus int requestType string requestPath string requestBody io.Reader expectedArray bool expectedPeer *nbpeer.Peer }{ { name: "GetPeersMetaData", requestType: http.MethodGet, requestPath: "/api/peers/", expectedStatus: http.StatusOK, expectedArray: true, expectedPeer: peer, }, { name: "GetPeer with update channel", requestType: http.MethodGet, requestPath: "/api/peers/" + testPeerID, expectedStatus: http.StatusOK, expectedArray: false, expectedPeer: peer, }, { name: "GetPeer with no update channel", requestType: http.MethodGet, requestPath: "/api/peers/" + peer1.ID, expectedStatus: http.StatusOK, expectedArray: false, expectedPeer: expectedPeer1, }, { name: "PutPeer", requestType: http.MethodPut, requestPath: "/api/peers/" + testPeerID, expectedStatus: http.StatusOK, expectedArray: false, requestBody: bytes.NewBufferString("{\"login_expiration_enabled\":true,\"name\":\"New Name\",\"ssh_enabled\":true}"), expectedPeer: expectedUpdatedPeer, }, } rr := httptest.NewRecorder() p := initTestMetaData(peer, peer1) for _, tc := range tt { t.Run(tc.name, func(t *testing.T) { recorder := httptest.NewRecorder() req := httptest.NewRequest(tc.requestType, tc.requestPath, tc.requestBody) ctx := context.WithValue(context.Background(), userIDKey, "admin_user") req = req.WithContext(ctx) router := mux.NewRouter() router.HandleFunc("/api/peers/", p.GetAllPeers).Methods("GET") router.HandleFunc("/api/peers/{peerId}", p.HandlePeer).Methods("GET") router.HandleFunc("/api/peers/{peerId}", p.HandlePeer).Methods("PUT") router.ServeHTTP(recorder, req) res := recorder.Result() defer res.Body.Close() if status := rr.Code; status != tc.expectedStatus { t.Fatalf("handler returned wrong status code: got %v want %v", status, http.StatusOK) } content, err := io.ReadAll(res.Body) if err != nil { t.Fatalf("I don't know what I expected; %v", err) } var got *api.Peer if tc.expectedArray { respBody := []*api.Peer{} err = json.Unmarshal(content, &respBody) if err != nil { t.Fatalf("Sent content is not in correct json format; %v", err) } // hardcode this check for now as we only have two peers in this suite assert.Equal(t, len(respBody), 2) for _, peer := range respBody { if peer.Id == testPeerID { got = peer } else { assert.Equal(t, peer.Connected, false) } } } else { got = &api.Peer{} err = json.Unmarshal(content, got) if err != nil { t.Fatalf("Sent content is not in correct json format; %v", err) } } t.Log(got) assert.Equal(t, got.Name, tc.expectedPeer.Name) assert.Equal(t, got.Version, tc.expectedPeer.Meta.WtVersion) assert.Equal(t, got.Ip, tc.expectedPeer.IP.String()) assert.Equal(t, got.Os, "OS core") assert.Equal(t, got.LoginExpirationEnabled, tc.expectedPeer.LoginExpirationEnabled) assert.Equal(t, got.SshEnabled, tc.expectedPeer.SSHEnabled) assert.Equal(t, got.Connected, tc.expectedPeer.Status.Connected) assert.Equal(t, got.SerialNumber, tc.expectedPeer.Meta.SystemSerialNumber) }) } } func TestGetAccessiblePeers(t *testing.T) { peer1 := &nbpeer.Peer{ ID: "peer1", Key: "key1", IP: net.ParseIP("100.64.0.1"), Status: &nbpeer.PeerStatus{Connected: true}, Name: "peer1", LoginExpirationEnabled: false, UserID: regularUser, } peer2 := &nbpeer.Peer{ ID: "peer2", Key: "key2", IP: net.ParseIP("100.64.0.2"), Status: &nbpeer.PeerStatus{Connected: true}, Name: "peer2", LoginExpirationEnabled: false, UserID: adminUser, } peer3 := &nbpeer.Peer{ ID: "peer3", Key: "key3", IP: net.ParseIP("100.64.0.3"), Status: &nbpeer.PeerStatus{Connected: true}, Name: "peer3", LoginExpirationEnabled: false, UserID: regularUser, } p := initTestMetaData(peer1, peer2, peer3) tt := []struct { name string peerID string callerUserID string expectedStatus int expectedPeers []string }{ { name: "non admin user can access owned peer", peerID: "peer1", callerUserID: regularUser, expectedStatus: http.StatusOK, expectedPeers: []string{"peer2", "peer3"}, }, { name: "non admin user can't access unowned peer", peerID: "peer2", callerUserID: regularUser, expectedStatus: http.StatusOK, expectedPeers: []string{}, }, { name: "admin user can access owned peer", peerID: "peer2", callerUserID: adminUser, expectedStatus: http.StatusOK, expectedPeers: []string{"peer1", "peer3"}, }, { name: "admin user can access unowned peer", peerID: "peer3", callerUserID: adminUser, expectedStatus: http.StatusOK, expectedPeers: []string{"peer1", "peer2"}, }, { name: "service user can access unowned peer", peerID: "peer3", callerUserID: serviceUser, expectedStatus: http.StatusOK, expectedPeers: []string{"peer1", "peer2"}, }, } for _, tc := range tt { t.Run(tc.name, func(t *testing.T) { recorder := httptest.NewRecorder() req := httptest.NewRequest(http.MethodGet, fmt.Sprintf("/api/peers/%s/accessible-peers", tc.peerID), nil) ctx := context.WithValue(context.Background(), userIDKey, tc.callerUserID) req = req.WithContext(ctx) router := mux.NewRouter() router.HandleFunc("/api/peers/{peerId}/accessible-peers", p.GetAccessiblePeers).Methods("GET") router.ServeHTTP(recorder, req) res := recorder.Result() if res.StatusCode != tc.expectedStatus { t.Fatalf("handler returned wrong status code: got %v want %v", res.StatusCode, tc.expectedStatus) } body, err := io.ReadAll(res.Body) if err != nil { t.Fatalf("failed to read response body: %v", err) } defer res.Body.Close() var accessiblePeers []api.AccessiblePeer err = json.Unmarshal(body, &accessiblePeers) if err != nil { t.Fatalf("failed to unmarshal response: %v", err) } peerIDs := make([]string, len(accessiblePeers)) for i, peer := range accessiblePeers { peerIDs[i] = peer.Id } assert.ElementsMatch(t, peerIDs, tc.expectedPeers) }) } }