mirror of
https://github.com/netbirdio/netbird.git
synced 2024-12-12 18:00:49 +01:00
Merge branch 'main' into feature/signal-dispatcher
This commit is contained in:
commit
adf55b3d93
@ -17,7 +17,7 @@
|
|||||||
<img src="https://img.shields.io/badge/license-BSD--3-blue" />
|
<img src="https://img.shields.io/badge/license-BSD--3-blue" />
|
||||||
</a>
|
</a>
|
||||||
<br>
|
<br>
|
||||||
<a href="https://join.slack.com/t/netbirdio/shared_invite/zt-vrahf41g-ik1v7fV8du6t0RwxSrJ96A">
|
<a href="https://join.slack.com/t/netbirdio/shared_invite/zt-2p5zwhm4g-8fHollzrQa5y4PZF5AEpvQ">
|
||||||
<img src="https://img.shields.io/badge/slack-@netbird-red.svg?logo=slack"/>
|
<img src="https://img.shields.io/badge/slack-@netbird-red.svg?logo=slack"/>
|
||||||
</a>
|
</a>
|
||||||
</p>
|
</p>
|
||||||
|
@ -3,6 +3,7 @@ package auth
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -180,7 +181,7 @@ func (d *DeviceAuthorizationFlow) WaitToken(ctx context.Context, info AuthFlowIn
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
return TokenInfo{}, fmt.Errorf(tokenResponse.ErrorDescription)
|
return TokenInfo{}, errors.New(tokenResponse.ErrorDescription)
|
||||||
}
|
}
|
||||||
|
|
||||||
tokenInfo := TokenInfo{
|
tokenInfo := TokenInfo{
|
||||||
|
@ -960,9 +960,9 @@ func (e *Engine) connWorker(conn *peer.Conn, peerKey string) {
|
|||||||
for {
|
for {
|
||||||
|
|
||||||
// randomize starting time a bit
|
// randomize starting time a bit
|
||||||
min := 500
|
minValue := 500
|
||||||
max := 2000
|
maxValue := 2000
|
||||||
duration := time.Duration(rand.Intn(max-min)+min) * time.Millisecond
|
duration := time.Duration(rand.Intn(maxValue-minValue)+minValue) * time.Millisecond
|
||||||
select {
|
select {
|
||||||
case <-e.ctx.Done():
|
case <-e.ctx.Done():
|
||||||
return
|
return
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// go:build !android
|
//go:build !android
|
||||||
|
|
||||||
package sysctl
|
package sysctl
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
@ -118,9 +118,9 @@ func (srv *DefaultServer) publicKeyHandler(ctx ssh.Context, key ssh.PublicKey) b
|
|||||||
|
|
||||||
func prepareUserEnv(user *user.User, shell string) []string {
|
func prepareUserEnv(user *user.User, shell string) []string {
|
||||||
return []string{
|
return []string{
|
||||||
fmt.Sprintf("SHELL=" + shell),
|
fmt.Sprint("SHELL=" + shell),
|
||||||
fmt.Sprintf("USER=" + user.Username),
|
fmt.Sprint("USER=" + user.Username),
|
||||||
fmt.Sprintf("HOME=" + user.HomeDir),
|
fmt.Sprint("HOME=" + user.HomeDir),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 9.0 KiB After Width: | Height: | Size: 9.6 KiB |
Binary file not shown.
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 12 KiB |
@ -2,6 +2,7 @@ package client
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"sync"
|
"sync"
|
||||||
@ -267,7 +268,7 @@ func (c *GrpcClient) receiveEvents(stream proto.ManagementService_SyncClient, se
|
|||||||
// GetServerPublicKey returns server's WireGuard public key (used later for encrypting messages sent to the server)
|
// GetServerPublicKey returns server's WireGuard public key (used later for encrypting messages sent to the server)
|
||||||
func (c *GrpcClient) GetServerPublicKey() (*wgtypes.Key, error) {
|
func (c *GrpcClient) GetServerPublicKey() (*wgtypes.Key, error) {
|
||||||
if !c.ready() {
|
if !c.ready() {
|
||||||
return nil, fmt.Errorf(errMsgNoMgmtConnection)
|
return nil, errors.New(errMsgNoMgmtConnection)
|
||||||
}
|
}
|
||||||
|
|
||||||
mgmCtx, cancel := context.WithTimeout(c.ctx, 5*time.Second)
|
mgmCtx, cancel := context.WithTimeout(c.ctx, 5*time.Second)
|
||||||
@ -314,7 +315,7 @@ func (c *GrpcClient) IsHealthy() bool {
|
|||||||
|
|
||||||
func (c *GrpcClient) login(serverKey wgtypes.Key, req *proto.LoginRequest) (*proto.LoginResponse, error) {
|
func (c *GrpcClient) login(serverKey wgtypes.Key, req *proto.LoginRequest) (*proto.LoginResponse, error) {
|
||||||
if !c.ready() {
|
if !c.ready() {
|
||||||
return nil, fmt.Errorf(errMsgNoMgmtConnection)
|
return nil, errors.New(errMsgNoMgmtConnection)
|
||||||
}
|
}
|
||||||
|
|
||||||
loginReq, err := encryption.EncryptMessage(serverKey, c.key, req)
|
loginReq, err := encryption.EncryptMessage(serverKey, c.key, req)
|
||||||
@ -452,7 +453,7 @@ func (c *GrpcClient) GetPKCEAuthorizationFlow(serverKey wgtypes.Key) (*proto.PKC
|
|||||||
// It should be used if there is changes on peer posture check after initial sync.
|
// It should be used if there is changes on peer posture check after initial sync.
|
||||||
func (c *GrpcClient) SyncMeta(sysInfo *system.Info) error {
|
func (c *GrpcClient) SyncMeta(sysInfo *system.Info) error {
|
||||||
if !c.ready() {
|
if !c.ready() {
|
||||||
return fmt.Errorf(errMsgNoMgmtConnection)
|
return errors.New(errMsgNoMgmtConnection)
|
||||||
}
|
}
|
||||||
|
|
||||||
serverPubKey, err := c.GetServerPublicKey()
|
serverPubKey, err := c.GetServerPublicKey()
|
||||||
|
@ -2072,6 +2072,28 @@ func (am *DefaultAccountManager) GetAccountIDForPeerKey(ctx context.Context, pee
|
|||||||
return am.Store.GetAccountIDByPeerPubKey(ctx, peerKey)
|
return am.Store.GetAccountIDByPeerPubKey(ctx, peerKey)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (am *DefaultAccountManager) handleUserPeer(ctx context.Context, peer *nbpeer.Peer, settings *Settings) (bool, error) {
|
||||||
|
user, err := am.Store.GetUserByUserID(ctx, peer.UserID)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = checkIfPeerOwnerIsBlocked(peer, user)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if peerLoginExpired(ctx, peer, settings) {
|
||||||
|
err = am.handleExpiredPeer(ctx, user, peer)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
// addAllGroup to account object if it doesn't exist
|
// addAllGroup to account object if it doesn't exist
|
||||||
func addAllGroup(account *Account) error {
|
func addAllGroup(account *Account) error {
|
||||||
if len(account.Groups) == 0 {
|
if len(account.Groups) == 0 {
|
||||||
|
@ -469,6 +469,35 @@ func (s *FileStore) GetUserByTokenID(_ context.Context, tokenID string) (*User,
|
|||||||
return account.Users[userID].Copy(), nil
|
return account.Users[userID].Copy(), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *FileStore) GetUserByUserID(_ context.Context, userID string) (*User, error) {
|
||||||
|
accountID, ok := s.UserID2AccountID[userID]
|
||||||
|
if !ok {
|
||||||
|
return nil, status.Errorf(status.NotFound, "accountID not found: provided userID doesn't exists")
|
||||||
|
}
|
||||||
|
|
||||||
|
account, err := s.getAccount(accountID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return account.Users[userID].Copy(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *FileStore) GetAccountGroups(ctx context.Context, accountID string) ([]*nbgroup.Group, error) {
|
||||||
|
account, err := s.getAccount(accountID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
groupsSlice := make([]*nbgroup.Group, 0, len(account.Groups))
|
||||||
|
|
||||||
|
for _, group := range account.Groups {
|
||||||
|
groupsSlice = append(groupsSlice, group)
|
||||||
|
}
|
||||||
|
|
||||||
|
return groupsSlice, nil
|
||||||
|
}
|
||||||
|
|
||||||
// GetAllAccounts returns all accounts
|
// GetAllAccounts returns all accounts
|
||||||
func (s *FileStore) GetAllAccounts(_ context.Context) (all []*Account) {
|
func (s *FileStore) GetAllAccounts(_ context.Context) (all []*Account) {
|
||||||
s.mux.Lock()
|
s.mux.Lock()
|
||||||
|
@ -257,7 +257,7 @@ func (s *GRPCServer) validateToken(ctx context.Context, jwtToken string) (string
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := s.accountManager.CheckUserAccessByJWTGroups(ctx, claims); err != nil {
|
if err := s.accountManager.CheckUserAccessByJWTGroups(ctx, claims); err != nil {
|
||||||
return "", status.Errorf(codes.PermissionDenied, err.Error())
|
return "", status.Error(codes.PermissionDenied, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
return claims.UserId, nil
|
return claims.UserId, nil
|
||||||
@ -268,15 +268,15 @@ func mapError(ctx context.Context, err error) error {
|
|||||||
if e, ok := internalStatus.FromError(err); ok {
|
if e, ok := internalStatus.FromError(err); ok {
|
||||||
switch e.Type() {
|
switch e.Type() {
|
||||||
case internalStatus.PermissionDenied:
|
case internalStatus.PermissionDenied:
|
||||||
return status.Errorf(codes.PermissionDenied, e.Message)
|
return status.Error(codes.PermissionDenied, e.Message)
|
||||||
case internalStatus.Unauthorized:
|
case internalStatus.Unauthorized:
|
||||||
return status.Errorf(codes.PermissionDenied, e.Message)
|
return status.Error(codes.PermissionDenied, e.Message)
|
||||||
case internalStatus.Unauthenticated:
|
case internalStatus.Unauthenticated:
|
||||||
return status.Errorf(codes.PermissionDenied, e.Message)
|
return status.Error(codes.PermissionDenied, e.Message)
|
||||||
case internalStatus.PreconditionFailed:
|
case internalStatus.PreconditionFailed:
|
||||||
return status.Errorf(codes.FailedPrecondition, e.Message)
|
return status.Error(codes.FailedPrecondition, e.Message)
|
||||||
case internalStatus.NotFound:
|
case internalStatus.NotFound:
|
||||||
return status.Errorf(codes.NotFound, e.Message)
|
return status.Error(codes.NotFound, e.Message)
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -46,7 +46,7 @@ func initPostureChecksTestData(postureChecks ...*posture.Checks) *PostureChecksH
|
|||||||
testPostureChecks[postureChecks.ID] = postureChecks
|
testPostureChecks[postureChecks.ID] = postureChecks
|
||||||
|
|
||||||
if err := postureChecks.Validate(); err != nil {
|
if err := postureChecks.Validate(); err != nil {
|
||||||
return status.Errorf(status.InvalidArgument, err.Error())
|
return status.Errorf(status.InvalidArgument, err.Error()) //nolint
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
@ -3,6 +3,7 @@ package idp
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -44,14 +45,14 @@ type mockJsonParser struct {
|
|||||||
|
|
||||||
func (m *mockJsonParser) Marshal(v interface{}) ([]byte, error) {
|
func (m *mockJsonParser) Marshal(v interface{}) ([]byte, error) {
|
||||||
if m.marshalErrorString != "" {
|
if m.marshalErrorString != "" {
|
||||||
return nil, fmt.Errorf(m.marshalErrorString)
|
return nil, errors.New(m.marshalErrorString)
|
||||||
}
|
}
|
||||||
return m.jsonParser.Marshal(v)
|
return m.jsonParser.Marshal(v)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *mockJsonParser) Unmarshal(data []byte, v interface{}) error {
|
func (m *mockJsonParser) Unmarshal(data []byte, v interface{}) error {
|
||||||
if m.unmarshalErrorString != "" {
|
if m.unmarshalErrorString != "" {
|
||||||
return fmt.Errorf(m.unmarshalErrorString)
|
return errors.New(m.unmarshalErrorString)
|
||||||
}
|
}
|
||||||
return m.jsonParser.Unmarshal(data, v)
|
return m.jsonParser.Unmarshal(data, v)
|
||||||
}
|
}
|
||||||
|
@ -150,7 +150,7 @@ func (m *JWTValidator) ValidateAndParse(ctx context.Context, token string) (*jwt
|
|||||||
// If we get here, the required token is missing
|
// If we get here, the required token is missing
|
||||||
errorMsg := "required authorization token not found"
|
errorMsg := "required authorization token not found"
|
||||||
log.WithContext(ctx).Debugf(" Error: No credentials found (CredentialsOptional=false)")
|
log.WithContext(ctx).Debugf(" Error: No credentials found (CredentialsOptional=false)")
|
||||||
return nil, fmt.Errorf(errorMsg)
|
return nil, errors.New(errorMsg)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now parse the token
|
// Now parse the token
|
||||||
@ -173,7 +173,7 @@ func (m *JWTValidator) ValidateAndParse(ctx context.Context, token string) (*jwt
|
|||||||
// Check if the parsed token is valid...
|
// Check if the parsed token is valid...
|
||||||
if !parsedToken.Valid {
|
if !parsedToken.Valid {
|
||||||
errorMsg := "token is invalid"
|
errorMsg := "token is invalid"
|
||||||
log.WithContext(ctx).Debugf(errorMsg)
|
log.WithContext(ctx).Debug(errorMsg)
|
||||||
return nil, errors.New(errorMsg)
|
return nil, errors.New(errorMsg)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -549,16 +549,25 @@ func (am *DefaultAccountManager) SyncPeer(ctx context.Context, sync PeerSync, ac
|
|||||||
return nil, nil, nil, status.NewPeerNotRegisteredError()
|
return nil, nil, nil, status.NewPeerNotRegisteredError()
|
||||||
}
|
}
|
||||||
|
|
||||||
err = checkIfPeerOwnerIsBlocked(peer, account)
|
if peer.UserID != "" {
|
||||||
if err != nil {
|
log.Infof("Peer has no userID")
|
||||||
return nil, nil, nil, err
|
|
||||||
|
user, err := account.FindUser(peer.UserID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = checkIfPeerOwnerIsBlocked(peer, user)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, nil, err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if peerLoginExpired(ctx, peer, account.Settings) {
|
if peerLoginExpired(ctx, peer, account.Settings) {
|
||||||
return nil, nil, nil, status.NewPeerLoginExpiredError()
|
return nil, nil, nil, status.NewPeerLoginExpiredError()
|
||||||
}
|
}
|
||||||
|
|
||||||
peer, updated := updatePeerMeta(peer, sync.Meta, account)
|
updated := peer.UpdateMetaIfNew(sync.Meta)
|
||||||
if updated {
|
if updated {
|
||||||
err = am.Store.SavePeer(ctx, account.Id, peer)
|
err = am.Store.SavePeer(ctx, account.Id, peer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -624,31 +633,28 @@ func (am *DefaultAccountManager) LoginPeer(ctx context.Context, login PeerLogin)
|
|||||||
// it means that the client has already checked if it needs login and had been through the SSO flow
|
// it means that the client has already checked if it needs login and had been through the SSO flow
|
||||||
// so, we can skip this check and directly proceed with the login
|
// so, we can skip this check and directly proceed with the login
|
||||||
if login.UserID == "" {
|
if login.UserID == "" {
|
||||||
|
log.Info("Peer needs login")
|
||||||
err = am.checkIFPeerNeedsLoginWithoutLock(ctx, accountID, login)
|
err = am.checkIFPeerNeedsLoginWithoutLock(ctx, accountID, login)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, nil, err
|
return nil, nil, nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unlock := am.Store.AcquireWriteLockByUID(ctx, accountID)
|
unlockAccount := am.Store.AcquireReadLockByUID(ctx, accountID)
|
||||||
|
defer unlockAccount()
|
||||||
|
unlockPeer := am.Store.AcquireWriteLockByUID(ctx, login.WireGuardPubKey)
|
||||||
defer func() {
|
defer func() {
|
||||||
if unlock != nil {
|
if unlockPeer != nil {
|
||||||
unlock()
|
unlockPeer()
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
// fetch the account from the store once more after acquiring lock to avoid concurrent updates inconsistencies
|
peer, err := am.Store.GetPeerByPeerPubKey(ctx, login.WireGuardPubKey)
|
||||||
account, err := am.Store.GetAccount(ctx, accountID)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, nil, err
|
return nil, nil, nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
peer, err := account.FindPeerByPubKey(login.WireGuardPubKey)
|
settings, err := am.Store.GetAccountSettings(ctx, accountID)
|
||||||
if err != nil {
|
|
||||||
return nil, nil, nil, status.NewPeerNotRegisteredError()
|
|
||||||
}
|
|
||||||
|
|
||||||
err = checkIfPeerOwnerIsBlocked(peer, account)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, nil, err
|
return nil, nil, nil, err
|
||||||
}
|
}
|
||||||
@ -656,21 +662,39 @@ func (am *DefaultAccountManager) LoginPeer(ctx context.Context, login PeerLogin)
|
|||||||
// this flag prevents unnecessary calls to the persistent store.
|
// this flag prevents unnecessary calls to the persistent store.
|
||||||
shouldStorePeer := false
|
shouldStorePeer := false
|
||||||
updateRemotePeers := false
|
updateRemotePeers := false
|
||||||
if peerLoginExpired(ctx, peer, account.Settings) {
|
|
||||||
err = am.handleExpiredPeer(ctx, login, account, peer)
|
if login.UserID != "" {
|
||||||
|
changed, err := am.handleUserPeer(ctx, peer, settings)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, nil, err
|
return nil, nil, nil, err
|
||||||
}
|
}
|
||||||
updateRemotePeers = true
|
if changed {
|
||||||
shouldStorePeer = true
|
shouldStorePeer = true
|
||||||
|
updateRemotePeers = true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
isRequiresApproval, isStatusChanged, err := am.integratedPeerValidator.IsNotValidPeer(ctx, account.Id, peer, account.GetPeerGroupsList(peer.ID), account.Settings.Extra)
|
groups, err := am.Store.GetAccountGroups(ctx, accountID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, nil, err
|
return nil, nil, nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
peer, updated := updatePeerMeta(peer, login.Meta, account)
|
var grps []string
|
||||||
|
for _, group := range groups {
|
||||||
|
for _, id := range group.Peers {
|
||||||
|
if id == peer.ID {
|
||||||
|
grps = append(grps, group.ID)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
isRequiresApproval, isStatusChanged, err := am.integratedPeerValidator.IsNotValidPeer(ctx, accountID, peer, grps, settings.Extra)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
updated := peer.UpdateMetaIfNew(login.Meta)
|
||||||
if updated {
|
if updated {
|
||||||
shouldStorePeer = true
|
shouldStorePeer = true
|
||||||
}
|
}
|
||||||
@ -687,8 +711,13 @@ func (am *DefaultAccountManager) LoginPeer(ctx context.Context, login PeerLogin)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unlock()
|
unlockPeer()
|
||||||
unlock = nil
|
unlockPeer = nil
|
||||||
|
|
||||||
|
account, err := am.Store.GetAccount(ctx, accountID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
if updateRemotePeers || isStatusChanged {
|
if updateRemotePeers || isStatusChanged {
|
||||||
am.updateAccountPeers(ctx, account)
|
am.updateAccountPeers(ctx, account)
|
||||||
@ -746,36 +775,30 @@ func (am *DefaultAccountManager) getValidatedPeerWithMap(ctx context.Context, is
|
|||||||
return peer, account.GetPeerNetworkMap(ctx, peer.ID, customZone, approvedPeersMap, am.metrics.AccountManagerMetrics()), postureChecks, nil
|
return peer, account.GetPeerNetworkMap(ctx, peer.ID, customZone, approvedPeersMap, am.metrics.AccountManagerMetrics()), postureChecks, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (am *DefaultAccountManager) handleExpiredPeer(ctx context.Context, login PeerLogin, account *Account, peer *nbpeer.Peer) error {
|
func (am *DefaultAccountManager) handleExpiredPeer(ctx context.Context, user *User, peer *nbpeer.Peer) error {
|
||||||
err := checkAuth(ctx, login.UserID, peer)
|
err := checkAuth(ctx, user.Id, peer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// If peer was expired before and if it reached this point, it is re-authenticated.
|
// If peer was expired before and if it reached this point, it is re-authenticated.
|
||||||
// UserID is present, meaning that JWT validation passed successfully in the API layer.
|
// UserID is present, meaning that JWT validation passed successfully in the API layer.
|
||||||
updatePeerLastLogin(peer, account)
|
peer = peer.UpdateLastLogin()
|
||||||
|
err = am.Store.SavePeer(ctx, peer.AccountID, peer)
|
||||||
// sync user last login with peer last login
|
|
||||||
user, err := account.FindUser(login.UserID)
|
|
||||||
if err != nil {
|
|
||||||
return status.Errorf(status.Internal, "couldn't find user")
|
|
||||||
}
|
|
||||||
|
|
||||||
err = am.Store.SaveUserLastLogin(account.Id, user.Id, peer.LastLogin)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
am.StoreEvent(ctx, login.UserID, peer.ID, account.Id, activity.UserLoggedInPeer, peer.EventMeta(am.GetDNSDomain()))
|
err = am.Store.SaveUserLastLogin(user.AccountID, user.Id, peer.LastLogin)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
am.StoreEvent(ctx, user.Id, peer.ID, user.AccountID, activity.UserLoggedInPeer, peer.EventMeta(am.GetDNSDomain()))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkIfPeerOwnerIsBlocked(peer *nbpeer.Peer, account *Account) error {
|
func checkIfPeerOwnerIsBlocked(peer *nbpeer.Peer, user *User) error {
|
||||||
if peer.AddedWithSSOLogin() {
|
if peer.AddedWithSSOLogin() {
|
||||||
user, err := account.FindUser(peer.UserID)
|
|
||||||
if err != nil {
|
|
||||||
return status.Errorf(status.PermissionDenied, "user doesn't exist")
|
|
||||||
}
|
|
||||||
if user.IsBlocked() {
|
if user.IsBlocked() {
|
||||||
return status.Errorf(status.PermissionDenied, "user is blocked")
|
return status.Errorf(status.PermissionDenied, "user is blocked")
|
||||||
}
|
}
|
||||||
@ -805,11 +828,6 @@ func peerLoginExpired(ctx context.Context, peer *nbpeer.Peer, settings *Settings
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func updatePeerLastLogin(peer *nbpeer.Peer, account *Account) {
|
|
||||||
peer.UpdateLastLogin()
|
|
||||||
account.UpdatePeer(peer)
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdatePeerSSHKey updates peer's public SSH key
|
// UpdatePeerSSHKey updates peer's public SSH key
|
||||||
func (am *DefaultAccountManager) UpdatePeerSSHKey(ctx context.Context, peerID string, sshKey string) error {
|
func (am *DefaultAccountManager) UpdatePeerSSHKey(ctx context.Context, peerID string, sshKey string) error {
|
||||||
if sshKey == "" {
|
if sshKey == "" {
|
||||||
@ -908,14 +926,6 @@ func (am *DefaultAccountManager) GetPeer(ctx context.Context, accountID, peerID,
|
|||||||
return nil, status.Errorf(status.Internal, "user %s has no access to peer %s under account %s", userID, peerID, accountID)
|
return nil, status.Errorf(status.Internal, "user %s has no access to peer %s under account %s", userID, peerID, accountID)
|
||||||
}
|
}
|
||||||
|
|
||||||
func updatePeerMeta(peer *nbpeer.Peer, meta nbpeer.PeerSystemMeta, account *Account) (*nbpeer.Peer, bool) {
|
|
||||||
if peer.UpdateMetaIfNew(meta) {
|
|
||||||
account.UpdatePeer(peer)
|
|
||||||
return peer, true
|
|
||||||
}
|
|
||||||
return peer, false
|
|
||||||
}
|
|
||||||
|
|
||||||
// updateAccountPeers updates all peers that belong to an account.
|
// updateAccountPeers updates all peers that belong to an account.
|
||||||
// Should be called when changes have to be synced to peers.
|
// Should be called when changes have to be synced to peers.
|
||||||
func (am *DefaultAccountManager) updateAccountPeers(ctx context.Context, account *Account) {
|
func (am *DefaultAccountManager) updateAccountPeers(ctx context.Context, account *Account) {
|
||||||
|
@ -60,7 +60,7 @@ func (am *DefaultAccountManager) SavePostureChecks(ctx context.Context, accountI
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := postureChecks.Validate(); err != nil {
|
if err := postureChecks.Validate(); err != nil {
|
||||||
return status.Errorf(status.InvalidArgument, err.Error())
|
return status.Errorf(status.InvalidArgument, err.Error()) //nolint
|
||||||
}
|
}
|
||||||
|
|
||||||
exists, uniqName := am.savePostureChecks(account, postureChecks)
|
exists, uniqName := am.savePostureChecks(account, postureChecks)
|
||||||
|
@ -468,6 +468,34 @@ func (s *SqlStore) GetUserByTokenID(ctx context.Context, tokenID string) (*User,
|
|||||||
return &user, nil
|
return &user, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *SqlStore) GetUserByUserID(ctx context.Context, userID string) (*User, error) {
|
||||||
|
var user User
|
||||||
|
result := s.db.First(&user, idQueryCondition, userID)
|
||||||
|
if result.Error != nil {
|
||||||
|
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
|
||||||
|
return nil, status.Errorf(status.NotFound, "user not found: index lookup failed")
|
||||||
|
}
|
||||||
|
log.WithContext(ctx).Errorf("error when getting user from the store: %s", result.Error)
|
||||||
|
return nil, status.Errorf(status.Internal, "issue getting user from store")
|
||||||
|
}
|
||||||
|
|
||||||
|
return &user, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SqlStore) GetAccountGroups(ctx context.Context, accountID string) ([]*nbgroup.Group, error) {
|
||||||
|
var groups []*nbgroup.Group
|
||||||
|
result := s.db.Find(&groups, idQueryCondition, accountID)
|
||||||
|
if result.Error != nil {
|
||||||
|
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
|
||||||
|
return nil, status.Errorf(status.NotFound, "accountID not found: index lookup failed")
|
||||||
|
}
|
||||||
|
log.WithContext(ctx).Errorf("error when getting groups from the store: %s", result.Error)
|
||||||
|
return nil, status.Errorf(status.Internal, "issue getting groups from store")
|
||||||
|
}
|
||||||
|
|
||||||
|
return groups, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *SqlStore) GetAllAccounts(ctx context.Context) (all []*Account) {
|
func (s *SqlStore) GetAllAccounts(ctx context.Context) (all []*Account) {
|
||||||
var accounts []Account
|
var accounts []Account
|
||||||
result := s.db.Find(&accounts)
|
result := s.db.Find(&accounts)
|
||||||
|
@ -41,6 +41,8 @@ type Store interface {
|
|||||||
GetAccountByPrivateDomain(ctx context.Context, domain string) (*Account, error)
|
GetAccountByPrivateDomain(ctx context.Context, domain string) (*Account, error)
|
||||||
GetTokenIDByHashedToken(ctx context.Context, secret string) (string, error)
|
GetTokenIDByHashedToken(ctx context.Context, secret string) (string, error)
|
||||||
GetUserByTokenID(ctx context.Context, tokenID string) (*User, error)
|
GetUserByTokenID(ctx context.Context, tokenID string) (*User, error)
|
||||||
|
GetUserByUserID(ctx context.Context, userID string) (*User, error)
|
||||||
|
GetAccountGroups(ctx context.Context, accountID string) ([]*nbgroup.Group, error)
|
||||||
GetPostureCheckByChecksDefinition(accountID string, checks *posture.ChecksDefinition) (*posture.Checks, error)
|
GetPostureCheckByChecksDefinition(accountID string, checks *posture.ChecksDefinition) (*posture.Checks, error)
|
||||||
SaveAccount(ctx context.Context, account *Account) error
|
SaveAccount(ctx context.Context, account *Account) error
|
||||||
SaveUsers(accountID string, users map[string]*User) error
|
SaveUsers(accountID string, users map[string]*User) error
|
||||||
|
@ -151,6 +151,22 @@ add_aur_repo() {
|
|||||||
${SUDO} pacman -Rs "$REMOVE_PKGS" --noconfirm
|
${SUDO} pacman -Rs "$REMOVE_PKGS" --noconfirm
|
||||||
}
|
}
|
||||||
|
|
||||||
|
prepare_tun_module() {
|
||||||
|
# Create the necessary file structure for /dev/net/tun
|
||||||
|
if [ ! -c /dev/net/tun ]; then
|
||||||
|
if [ ! -d /dev/net ]; then
|
||||||
|
mkdir -m 755 /dev/net
|
||||||
|
fi
|
||||||
|
mknod /dev/net/tun c 10 200
|
||||||
|
chmod 0755 /dev/net/tun
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Load the tun module if not already loaded
|
||||||
|
if ! lsmod | grep -q "^tun\s"; then
|
||||||
|
insmod /lib/modules/tun.ko
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
install_native_binaries() {
|
install_native_binaries() {
|
||||||
# Checks for supported architecture
|
# Checks for supported architecture
|
||||||
case "$ARCH" in
|
case "$ARCH" in
|
||||||
@ -268,6 +284,10 @@ install_netbird() {
|
|||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
|
if [ "$OS_NAME" = "synology" ]; then
|
||||||
|
prepare_tun_module
|
||||||
|
fi
|
||||||
|
|
||||||
# Add package manager to config
|
# Add package manager to config
|
||||||
${SUDO} mkdir -p "$CONFIG_FOLDER"
|
${SUDO} mkdir -p "$CONFIG_FOLDER"
|
||||||
echo "package_manager=$PACKAGE_MANAGER" | ${SUDO} tee "$CONFIG_FILE" > /dev/null
|
echo "package_manager=$PACKAGE_MANAGER" | ${SUDO} tee "$CONFIG_FILE" > /dev/null
|
||||||
|
@ -10,5 +10,5 @@ import (
|
|||||||
|
|
||||||
// Listen is not supported on other platforms then Linux
|
// Listen is not supported on other platforms then Linux
|
||||||
func Listen(port int, filter BPFFilter) (net.PacketConn, error) {
|
func Listen(port int, filter BPFFilter) (net.PacketConn, error) {
|
||||||
return nil, fmt.Errorf(fmt.Sprintf("Not supported OS %s. SharedSocket is only supported on Linux", runtime.GOOS))
|
return nil, fmt.Errorf("not supported OS %s. SharedSocket is only supported on Linux", runtime.GOOS)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user