Add a flag to imap settings to enable/disable oauth2 scheme

This commit is contained in:
Eike Kettner 2021-01-03 16:36:19 +01:00
parent ac6a7b28be
commit 0cfd8974d3
8 changed files with 69 additions and 10 deletions

View File

@ -123,7 +123,8 @@ object OMail {
imapUser: Option[String], imapUser: Option[String],
imapPassword: Option[Password], imapPassword: Option[Password],
imapSsl: SSLType, imapSsl: SSLType,
imapCertCheck: Boolean imapCertCheck: Boolean,
imapOAuth2: Boolean
) { ) {
def toRecord(accId: AccountId) = def toRecord(accId: AccountId) =
@ -135,7 +136,8 @@ object OMail {
imapUser, imapUser,
imapPassword, imapPassword,
imapSsl, imapSsl,
imapCertCheck imapCertCheck,
imapOAuth2
) )
} }

View File

@ -3873,6 +3873,7 @@ components:
- from - from
- sslType - sslType
- ignoreCertificates - ignoreCertificates
- useOAuth
properties: properties:
name: name:
type: string type: string
@ -3891,6 +3892,11 @@ components:
type: string type: string
ignoreCertificates: ignoreCertificates:
type: boolean type: boolean
useOAuth:
type: boolean
description: |
Use the password as an OAuth2 access token with the
authentication scheme XOAUTH2.
CalEventCheckResult: CalEventCheckResult:
description: | description: |
The result of checking a calendar event string. The result of checking a calendar event string.

View File

@ -166,7 +166,8 @@ object MailSettingsRoutes {
ru.imapUser, ru.imapUser,
ru.imapPassword, ru.imapPassword,
ru.imapSsl.name, ru.imapSsl.name,
!ru.imapCertCheck !ru.imapCertCheck,
ru.imapOAuth2
) )
def makeSmtpSettings(ems: EmailSettings): Either[String, OMail.SmtpSettings] = { def makeSmtpSettings(ems: EmailSettings): Either[String, OMail.SmtpSettings] = {
@ -203,6 +204,7 @@ object MailSettingsRoutes {
ims.imapUser, ims.imapUser,
ims.imapPassword, ims.imapPassword,
sslt, sslt,
!ims.ignoreCertificates !ims.ignoreCertificates,
ims.useOAuth
) )
} }

View File

@ -0,0 +1,7 @@
ALTER TABLE "userimap"
ADD COLUMN "imap_oauth2" boolean NULL;
UPDATE "userimap" SET "imap_oauth2" = false;
ALTER TABLE "userimap"
ALTER COLUMN "imap_oauth2" SET NOT NULL;

View File

@ -0,0 +1,7 @@
ALTER TABLE `userimap`
ADD COLUMN (`imap_oauth2` boolean);
UPDATE `userimap` SET `imap_oauth2` = false;
ALTER TABLE `userimap`
MODIFY `imap_oauth2` boolean NOT NULL;

View File

@ -0,0 +1,7 @@
ALTER TABLE "userimap"
ADD COLUMN "imap_oauth2" boolean NULL;
UPDATE "userimap" SET "imap_oauth2" = false;
ALTER TABLE "userimap"
ALTER COLUMN "imap_oauth2" SET NOT NULL;

View File

@ -22,6 +22,7 @@ case class RUserImap(
imapPassword: Option[Password], imapPassword: Option[Password],
imapSsl: SSLType, imapSsl: SSLType,
imapCertCheck: Boolean, imapCertCheck: Boolean,
imapOAuth2: Boolean,
created: Timestamp created: Timestamp
) { ) {
@ -32,6 +33,7 @@ case class RUserImap(
imapUser.getOrElse(""), imapUser.getOrElse(""),
imapPassword.map(_.pass).getOrElse(""), imapPassword.map(_.pass).getOrElse(""),
imapSsl, imapSsl,
imapOAuth2,
!imapCertCheck !imapCertCheck
) )
} }
@ -47,7 +49,8 @@ object RUserImap {
imapUser: Option[String], imapUser: Option[String],
imapPassword: Option[Password], imapPassword: Option[Password],
imapSsl: SSLType, imapSsl: SSLType,
imapCertCheck: Boolean imapCertCheck: Boolean,
imapOAuth2: Boolean
): F[RUserImap] = ): F[RUserImap] =
for { for {
now <- Timestamp.current[F] now <- Timestamp.current[F]
@ -62,6 +65,7 @@ object RUserImap {
imapPassword, imapPassword,
imapSsl, imapSsl,
imapCertCheck, imapCertCheck,
imapOAuth2,
now now
) )
@ -73,7 +77,8 @@ object RUserImap {
imapUser: Option[String], imapUser: Option[String],
imapPassword: Option[Password], imapPassword: Option[Password],
imapSsl: SSLType, imapSsl: SSLType,
imapCertCheck: Boolean imapCertCheck: Boolean,
imapOAuth2: Boolean
): OptionT[ConnectionIO, RUserImap] = ): OptionT[ConnectionIO, RUserImap] =
for { for {
now <- OptionT.liftF(Timestamp.current[ConnectionIO]) now <- OptionT.liftF(Timestamp.current[ConnectionIO])
@ -89,6 +94,7 @@ object RUserImap {
imapPassword, imapPassword,
imapSsl, imapSsl,
imapCertCheck, imapCertCheck,
imapOAuth2,
now now
) )
@ -104,6 +110,7 @@ object RUserImap {
val imapPass = Column[Password]("imap_password", this) val imapPass = Column[Password]("imap_password", this)
val imapSsl = Column[SSLType]("imap_ssl", this) val imapSsl = Column[SSLType]("imap_ssl", this)
val imapCertCheck = Column[Boolean]("imap_certcheck", this) val imapCertCheck = Column[Boolean]("imap_certcheck", this)
val imapOAuth2 = Column[Boolean]("imap_oauth2", this)
val created = Column[Timestamp]("created", this) val created = Column[Timestamp]("created", this)
val all = NonEmptyList.of[Column[_]]( val all = NonEmptyList.of[Column[_]](
@ -116,6 +123,7 @@ object RUserImap {
imapPass, imapPass,
imapSsl, imapSsl,
imapCertCheck, imapCertCheck,
imapOAuth2,
created created
) )
} }
@ -129,7 +137,7 @@ object RUserImap {
.insert( .insert(
t, t,
t.all, t.all,
sql"${v.id},${v.uid},${v.name},${v.imapHost},${v.imapPort},${v.imapUser},${v.imapPassword},${v.imapSsl},${v.imapCertCheck},${v.created}" sql"${v.id},${v.uid},${v.name},${v.imapHost},${v.imapPort},${v.imapUser},${v.imapPassword},${v.imapSsl},${v.imapCertCheck},${v.imapOAuth2},${v.created}"
) )
} }
@ -145,7 +153,8 @@ object RUserImap {
t.imapUser.setTo(v.imapUser), t.imapUser.setTo(v.imapUser),
t.imapPass.setTo(v.imapPassword), t.imapPass.setTo(v.imapPassword),
t.imapSsl.setTo(v.imapSsl), t.imapSsl.setTo(v.imapSsl),
t.imapCertCheck.setTo(v.imapCertCheck) t.imapCertCheck.setTo(v.imapCertCheck),
t.imapOAuth2.setTo(v.imapOAuth2)
) )
) )
} }

View File

@ -32,6 +32,7 @@ type alias Model =
, password : Maybe String , password : Maybe String
, sslType : Comp.Dropdown.Model SSLType , sslType : Comp.Dropdown.Model SSLType
, ignoreCertificates : Bool , ignoreCertificates : Bool
, useOAuthToken : Bool
} }
@ -58,6 +59,7 @@ emptyModel =
, selected = Just Data.SSLType.None , selected = Just Data.SSLType.None
} }
, ignoreCertificates = False , ignoreCertificates = False
, useOAuthToken = False
} }
@ -87,6 +89,7 @@ init ems =
|> Just |> Just
} }
, ignoreCertificates = ems.ignoreCertificates , ignoreCertificates = ems.ignoreCertificates
, useOAuthToken = ems.useOAuth
} }
@ -104,6 +107,7 @@ getSettings model =
|> Maybe.withDefault Data.SSLType.None |> Maybe.withDefault Data.SSLType.None
|> Data.SSLType.toString |> Data.SSLType.toString
, ignoreCertificates = model.ignoreCertificates , ignoreCertificates = model.ignoreCertificates
, useOAuth = model.useOAuthToken
} }
) )
@ -116,6 +120,7 @@ type Msg
| PassMsg Comp.PasswordInput.Msg | PassMsg Comp.PasswordInput.Msg
| SSLTypeMsg (Comp.Dropdown.Msg SSLType) | SSLTypeMsg (Comp.Dropdown.Msg SSLType)
| ToggleCheckCert | ToggleCheckCert
| ToggleUseOAuth
isValid : Model -> Bool isValid : Model -> Bool
@ -159,14 +164,17 @@ update msg model =
ToggleCheckCert -> ToggleCheckCert ->
( { model | ignoreCertificates = not model.ignoreCertificates }, Cmd.none ) ( { model | ignoreCertificates = not model.ignoreCertificates }, Cmd.none )
ToggleUseOAuth ->
( { model | useOAuthToken = not model.useOAuthToken }, Cmd.none )
view : UiSettings -> Model -> Html Msg view : UiSettings -> Model -> Html Msg
view settings model = view settings model =
div div
[ classList [ classList
[ ( "ui form", True ) [ ( "ui form", True )
, ( "error", not (isValid model) ) , ( "info error", not (isValid model) )
, ( "success", isValid model ) , ( "info success", isValid model )
] ]
] ]
[ div [ class "required field" ] [ div [ class "required field" ]
@ -227,6 +235,17 @@ view settings model =
, label [] [ text "Ignore certificate check" ] , label [] [ text "Ignore certificate check" ]
] ]
] ]
, div [ class "inline field" ]
[ div [ class "ui checkbox" ]
[ input
[ type_ "checkbox"
, checked model.useOAuthToken
, onCheck (\_ -> ToggleUseOAuth)
]
[]
, label [] [ text "Enable OAuth2 authentication using the password as access token" ]
]
]
] ]
, div [ class "two fields" ] , div [ class "two fields" ]
[ div [ class "field" ] [ div [ class "field" ]