Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow displaying emails of users #719

Closed
wants to merge 11 commits into from
3 changes: 3 additions & 0 deletions libs/brig-types/src/Brig/Types/Swagger.hs
Original file line number Diff line number Diff line change
Expand Up @@ -141,6 +141,9 @@ user = defineModel "User" $ do
description "User ID"
property "name" string' $
description "Name"
property "email" string' $ do
description "Email"
optional
property "assets" (array (ref asset)) $
description "Profile assets"
property "accent_id" int32' $ do
Expand Down
63 changes: 60 additions & 3 deletions libs/brig-types/src/Brig/Types/User.hs
Original file line number Diff line number Diff line change
Expand Up @@ -103,12 +103,41 @@ connectedProfile u = UserProfile
, profileDeleted = userDeleted u
, profileExpire = userExpire u
, profileTeam = userTeam u
-- We don't want to show the email by default;
-- However we do allow adding it back in intentionally later.
, profileEmail = Nothing
}

publicProfile :: User -> UserProfile
publicProfile u = (connectedProfile u)
{ profileLocale = Nothing
}
publicProfile u =
-- Note that we explicitly unpack and repack the types here rather than using
-- RecordWildCards or something similar because we want changes to the public profile
-- to be EXPLICIT and INTENTIONAL so we don't accidentally leak sensitive data.
let UserProfile { profileId
, profileHandle
, profileName
, profilePict
, profileAssets
, profileAccentId
, profileService
, profileDeleted
, profileExpire
, profileTeam
} = connectedProfile u
in UserProfile
{ profileLocale = Nothing
, profileEmail = Nothing
, profileId
, profileHandle
, profileName
, profilePict
, profileAssets
, profileAccentId
, profileService
, profileDeleted
, profileExpire
, profileTeam
}

-- | The data of an existing user.
data User = User
Expand All @@ -134,6 +163,31 @@ data User = User
}
deriving (Eq, Show)


-- | Configurations for whether to show a user's email to others.
data EmailVisibility
= EmailVisibleIfOnTeam
{- ^ Anyone on a team can see the email of anyone else who is on a team.
Regardless of if they're on the SAME team.
This may sound strange; but certain on-premise hosters have many different teams
and still want them to see each-other's emails.
-}
| EmailVisibleToSelf
-- ^ Never show emails to anyone other than yourself
deriving (Eq, Show)

instance FromJSON EmailVisibility where
parseJSON = withText "EmailVisibility" $ \case
"visible_if_on_team" -> pure EmailVisibleIfOnTeam
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This name isn't super clear about its behaviour. Anyone have suggestions?

"visible_to_self" -> pure EmailVisibleToSelf
_ -> fail
$ "unexpected value for EmailVisibility settings: "
<> "expected one of [visible_if_on_team, visible_to_same_team, visible_to_self]"

instance ToJSON EmailVisibility where
ChrisPenner marked this conversation as resolved.
Show resolved Hide resolved
toJSON EmailVisibleIfOnTeam = "visible_if_on_team"
toJSON EmailVisibleToSelf = "visible_to_self"

userEmail :: User -> Maybe Email
userEmail = emailIdentity <=< userIdentity

Expand All @@ -160,6 +214,7 @@ data UserProfile = UserProfile
, profileLocale :: !(Maybe Locale)
, profileExpire :: !(Maybe UTCTimeMillis)
, profileTeam :: !(Maybe TeamId)
, profileEmail :: !(Maybe Email)
}
deriving (Eq, Show)

Expand Down Expand Up @@ -213,6 +268,7 @@ instance FromJSON UserProfile where
<*> o .:? "locale"
<*> o .:? "expires_at"
<*> o .:? "team"
<*> o .:? "email"

instance ToJSON UserProfile where
toJSON u = object
Expand All @@ -227,6 +283,7 @@ instance ToJSON UserProfile where
# "locale" .= profileLocale u
# "expires_at" .= profileExpire u
# "team" .= profileTeam u
# "email" .= profileEmail u
# []

instance FromJSON SelfProfile where
Expand Down
2 changes: 2 additions & 0 deletions services/brig/brig.integration.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -141,6 +141,8 @@ optSettings:
setDefaultLocale: en
setMaxTeamSize: 32
setMaxConvSize: 16
optMutableSettings:
Copy link
Contributor Author

@ChrisPenner ChrisPenner Apr 15, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there a better name for this? It's not great that our testing strategy leaks into our config 😢

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

One more way to test this would be to create a brig Application with a different config, and test against that. done here

setEmailVisibility: visible_to_self

logLevel: Warn
logNetStrings: false
1 change: 1 addition & 0 deletions services/brig/package.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ ghc-options:
- -funbox-strict-fields
dependencies:
- aeson >=0.11
- barbies
- base ==4.*
- base64-bytestring >=1.0
- bloodhound >=0.13
Expand Down
7 changes: 7 additions & 0 deletions services/brig/src/Brig/API.hs
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ import qualified Brig.API.Client as API
import qualified Brig.API.Connection as API
import qualified Brig.API.Properties as API
import qualified Brig.API.User as API
import qualified Brig.API.Settings as API
import qualified Brig.Data.User as Data
import qualified Brig.Team.Util as Team
import qualified Brig.User.API.Auth as Auth
Expand Down Expand Up @@ -73,6 +74,12 @@ sitemap o = do
get "/i/monitoring" (continue $ const $ view metrics >>= fmap json . render) $
accept "application" "json"

get "/i/settings" (continue $ API.getSettings) $
accept "application" "json"

put "/i/settings" (continue $ API.putSettings) $
jsonRequest @(MutableSettings' Maybe)

post "/i/users/:id/auto-connect" (continue autoConnect) $
accept "application" "json"
.&. capture "id"
Expand Down
38 changes: 38 additions & 0 deletions services/brig/src/Brig/API/Settings.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
module Brig.API.Settings
( putSettings
, getSettings
) where

import Imports

import Brig.API.Handler (Handler, JSON)
import Brig.App (mutableSettings)
import Brig.Options (MutableSettings, MutableSettings')
import Control.Lens
import Data.Barbie (bzipWith)
import Network.HTTP.Types.Status (status200)
import Network.Wai (Response)
import Network.Wai.Utilities (JsonRequest, empty, json, parseBody', setStatus)

-- | Update the provided settings accordingly
putSettings :: JsonRequest (MutableSettings' Maybe) -> Handler Response
putSettings body = do
newSettings <- parseBody' body
mSettingsVar <- view mutableSettings
atomically $ do
mSettings <- readTVar mSettingsVar
let mergedSettings :: MutableSettings
mergedSettings = bzipWith fromMaybe' mSettings newSettings
writeTVar mSettingsVar mergedSettings
return $ (setStatus status200 empty)
where
fromMaybe' :: Identity a -> Maybe a -> Identity a
fromMaybe' a ma = maybe a Identity ma


-- | Update the provided settings accordingly
getSettings :: JSON -> Handler Response
getSettings _ = do
mSet <- view mutableSettings >>= readTVarIO
return . setStatus status200
$ json mSet
25 changes: 20 additions & 5 deletions services/brig/src/Brig/API/User.hs
Original file line number Diff line number Diff line change
Expand Up @@ -837,14 +837,29 @@ lookupProfiles :: UserId -- ^ User 'A' on whose behalf the profiles are reques
lookupProfiles self others = do
users <- Data.lookupUsers others >>= mapM userGC
css <- toMap <$> Data.lookupConnectionStatus (map userId users) [self]
return $ map (toProfile css) users
Identity emailVisibility <- view mutableSettings >>= readTVarIO >>= pure . setEmailVisibility
return $ map (toProfile emailVisibility css) users
where
toMap :: [ConnectionStatus] -> Map UserId Relation
toMap = Map.fromList . map (csFrom &&& csStatus)
toProfile css u =
toProfile :: EmailVisibility -> Map UserId Relation -> User -> UserProfile
toProfile emailVisibility css u =
let cs = Map.lookup (userId u) css
in if userId u == self || cs == Just Accepted || cs == Just Sent
then connectedProfile u
else publicProfile u
profileEmail' = getEmailForProfile u emailVisibility
baseProfile = if userId u == self || cs == Just Accepted || cs == Just Sent
then connectedProfile u
else publicProfile u
in baseProfile{ profileEmail = profileEmail'}

-- | Gets the email if it's visible to the requester according to configured settings
getEmailForProfile :: User -- ^ The user who's profile is being requested
-> EmailVisibility
-> Maybe Email
getEmailForProfile _ EmailVisibleToSelf = Nothing
getEmailForProfile u EmailVisibleIfOnTeam =
if isJust (userTeam u)
then userEmail u
else Nothing

-- | Obtain a profile for a user as he can see himself.
lookupSelfProfile :: UserId -> AppIO (Maybe SelfProfile)
Expand Down
62 changes: 33 additions & 29 deletions services/brig/src/Brig/App.hs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ module Brig.App
, nexmoCreds
, twilioCreds
, settings
, mutableSettings
, currentTime
, geoDb
, zauthEnv
Expand Down Expand Up @@ -134,6 +135,7 @@ data Env = Env
, _httpManager :: Manager
, _extGetManager :: (Manager, [Fingerprint Rsa] -> SSL.SSL -> IO ())
, _settings :: Settings
, _mutableSettings :: TVar Opt.MutableSettings
, _nexmoCreds :: Nexmo.Credentials
, _twilioCreds :: Twilio.Credentials
, _geoDb :: Maybe (IORef GeoIp.GeoDB)
Expand Down Expand Up @@ -172,6 +174,7 @@ newEnv o = do
g <- geoSetup lgr w $ Opt.geoDb o
(turn, turnV2) <- turnSetup lgr w sha512 (Opt.turn o)
let sett = Opt.optSettings o
mSett <- newTVarIO $ Opt.optMutableSettings o
nxm <- initCredentials (Opt.setNexmo sett)
twl <- initCredentials (Opt.setTwilio sett)
stomp <- case (Opt.stomp o, Opt.setStomp sett) of
Expand All @@ -185,35 +188,36 @@ newEnv o = do
StompQueue q -> pure (StompQueue q)
SqsQueue q -> SqsQueue <$> AWS.getQueueUrl (aws ^. AWS.amazonkaEnv) q
return $! Env
{ _cargohold = mkEndpoint $ Opt.cargohold o
, _galley = mkEndpoint $ Opt.galley o
, _gundeck = mkEndpoint $ Opt.gundeck o
, _casClient = cas
, _smtpEnv = emailSMTP
, _awsEnv = aws
, _stompEnv = stomp
, _metrics = mtr
, _applog = lgr
, _internalEvents = eventsQueue
, _requestId = def
, _usrTemplates = utp
, _provTemplates = ptp
, _tmTemplates = ttp
, _templateBranding = branding
, _httpManager = mgr
, _extGetManager = ext
, _settings = sett
, _nexmoCreds = nxm
, _twilioCreds = twl
, _geoDb = g
, _turnEnv = turn
, _turnEnvV2 = turnV2
, _fsWatcher = w
, _currentTime = clock
, _zauthEnv = zau
, _digestMD5 = md5
, _digestSHA256 = sha256
, _indexEnv = mkIndexEnv o lgr mgr mtr
{ _cargohold = mkEndpoint $ Opt.cargohold o
, _galley = mkEndpoint $ Opt.galley o
, _gundeck = mkEndpoint $ Opt.gundeck o
, _casClient = cas
, _smtpEnv = emailSMTP
, _awsEnv = aws
, _stompEnv = stomp
, _metrics = mtr
, _applog = lgr
, _internalEvents = eventsQueue
, _requestId = def
, _usrTemplates = utp
, _provTemplates = ptp
, _tmTemplates = ttp
, _templateBranding = branding
, _httpManager = mgr
, _extGetManager = ext
, _settings = sett
, _mutableSettings = mSett
ChrisPenner marked this conversation as resolved.
Show resolved Hide resolved
, _nexmoCreds = nxm
, _twilioCreds = twl
, _geoDb = g
, _turnEnv = turn
, _turnEnvV2 = turnV2
, _fsWatcher = w
, _currentTime = clock
, _zauthEnv = zau
, _digestMD5 = md5
, _digestSHA256 = sha256
, _indexEnv = mkIndexEnv o lgr mgr mtr
}
where
emailConn _ (Opt.EmailAWS aws) = return (Just aws, Nothing)
Expand Down
Loading