Skip to content

Commit

Permalink
Merge pull request #462 from ForgeRock/SDKS-3423
Browse files Browse the repository at this point in the history
SDKS-3423 Handle Sign out with ID Token for PingOne Platform.
  • Loading branch information
spetrov authored Nov 18, 2024
2 parents 1ad6a27 + a57b166 commit 0cb37c3
Show file tree
Hide file tree
Showing 5 changed files with 216 additions and 117 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
## [4.X.X]
#### Added
- Self device management [SDKS-3408]
- Support Sign out with ID Token for PingOne Platform [SDKS-3423]

## [4.6.0]
#### Added
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -105,8 +105,13 @@ data class FROptions(val server: Server,
tokenEndpoint = openIdConfiguration?.getString("token_endpoint"),
userinfoEndpoint = openIdConfiguration?.getString("userinfo_endpoint"),
revokeEndpoint = openIdConfiguration?.getString("revocation_endpoint"),
endSessionEndpoint = openIdConfiguration?.getString("end_session_endpoint"))

endSessionEndpoint = if (oauth.oauthSignOutRedirectUri.isEmpty()) {
openIdConfiguration?.optString("ping_end_idp_session_endpoint",
openIdConfiguration.getString("end_session_endpoint"))
} else {
openIdConfiguration?.getString("end_session_endpoint")
}
)
this@FROptions.copy(urlPath = urlPath, server = server)

}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,7 @@ class OAuth2Client(
try {
request = Request.Builder()
.url(getEndSessionUrl(clientId, idToken))
.header("Accept", "application/json")
.get()
.tag(END_SESSION)
.build()
Expand Down
291 changes: 176 additions & 115 deletions forgerock-auth/src/test/java/org/forgerock/android/auth/FROptionTest.kt
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
package org.forgerock.android.auth

import androidx.test.ext.junit.runners.AndroidJUnit4
import kotlinx.coroutines.runBlocking
import kotlinx.coroutines.test.runTest
import okhttp3.OkHttpClient
import okhttp3.mockwebserver.MockResponse
import org.forgerock.android.auth.exception.ApiException
Expand All @@ -19,19 +19,25 @@ import org.junit.runner.RunWith
import java.net.HttpURLConnection

@RunWith(AndroidJUnit4::class)
class FROptionTest: BaseTest() {
class FROptionTest : BaseTest() {
@Test
fun testDefaultBuilderOption() {
class TestCustomLogger: FRLogger {
override fun error(tag: String?, t: Throwable?, message: String?, vararg values: Any?) {}
class TestCustomLogger : FRLogger {
override fun error(tag: String?,
t: Throwable?,
message: String?,
vararg values: Any?) {
}

override fun error(tag: String?, message: String?, vararg values: Any?) {}
override fun warn(tag: String?, message: String?, vararg values: Any?) {}
override fun warn(tag: String?, t: Throwable?, message: String?, vararg values: Any?) {}
override fun debug(tag: String?, message: String?, vararg values: Any?) {}
override fun info(tag: String?, message: String?, vararg values: Any?) {}
override fun network(tag: String?, message: String?, vararg values: Any?) {}
}
val logger:FRLogger = TestCustomLogger()

val logger: FRLogger = TestCustomLogger()
val option = FROptionsBuilder.build {
server {
url = "https://andy.com"
Expand Down Expand Up @@ -69,150 +75,205 @@ class FROptionTest: BaseTest() {

@Test(expected = IllegalArgumentException::class)
fun testInValidConfigRealm() {
val option1 = FROptionsBuilder.build { server {
url = "https://stoyan.com"
realm = ""
}}
val option1 = FROptionsBuilder.build {
server {
url = "https://stoyan.com"
realm = ""
}
}
option1.validateConfig()
}

@Test(expected = IllegalArgumentException::class)
fun testInValidConfigCookieName() {
val option1 = FROptionsBuilder.build { server {
url = "https://stoyan.com"
cookieName = ""
}}
val option1 = FROptionsBuilder.build {
server {
url = "https://stoyan.com"
cookieName = ""
}
}
option1.validateConfig()
}

@Test(expected = IllegalArgumentException::class)
fun testInValidConfigUrl() {
val option1 = FROptionsBuilder.build { server {
url = ""
}}
val option1 = FROptionsBuilder.build {
server {
url = ""
}
}
option1.validateConfig()
}

@Test
fun testReferenceAndValue() {
val option1 = FROptionsBuilder.build { server {
url = "https://stoyan.com"
}}
var option2 = FROptionsBuilder.build { server {
url = "https://stoyan.com"
}}
val option1 = FROptionsBuilder.build {
server {
url = "https://stoyan.com"
}
}
var option2 = FROptionsBuilder.build {
server {
url = "https://stoyan.com"
}
}
assertTrue(FROptions.equals(option1, option2))
option2 = FROptionsBuilder.build { server {
url = "https://andy.com"
realm = ""
}}
option2 = FROptionsBuilder.build {
server {
url = "https://andy.com"
realm = ""
}
}
assertFalse(FROptions.equals(option1, option2))
assertTrue(FROptions.equals(option2, option2))
}

@Test
fun testDiscoverEndpointFailure() {
runBlocking {
val option1 = FROptionsBuilder.build {
server {
url = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as"
}
urlPath {
authenticateEndpoint = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate"
sessionEndpoint = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session"
}
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
}
}

server.enqueue(MockResponse().setResponseCode(HttpURLConnection.HTTP_FORBIDDEN))

try {
val copied =
option1.discover(url)
FRAuth.start(context, copied)
fail()
}
catch (e: ApiException) {
assertTrue(e.statusCode == HttpURLConnection.HTTP_FORBIDDEN)
fun testDiscoverEndpointFailure() = runTest {
val option1 = FROptionsBuilder.build {
server {
url = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as"
}
urlPath {
authenticateEndpoint =
"https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate"
sessionEndpoint =
"https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session"
}
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
}
}

server.enqueue(MockResponse().setResponseCode(HttpURLConnection.HTTP_FORBIDDEN))

try {
val copied =
option1.discover(url)
FRAuth.start(context, copied)
fail()
} catch (e: ApiException) {
assertTrue(e.statusCode == HttpURLConnection.HTTP_FORBIDDEN)
}
}

@Test
fun testDiscoverEndpointWithURLProvided() = runTest {
server.enqueue(
MockResponse()
.setResponseCode(HttpURLConnection.HTTP_OK)
.addHeader("Content-Type", "application/json")
.setBody(getJson("/discovery.json")))

val option1 = FROptionsBuilder.build {
server {
url = "https://auth.pingone.us/02fb4743-189a-4bc7-9d6c-a919edfe6447/as"
}
urlPath {
authenticateEndpoint =
"https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate"
sessionEndpoint =
"https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session"
}
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
}
}
val copied = option1.discover(url)
assertTrue(copied.urlPath.authorizeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authorize")
assertTrue(copied.server.url == "https://auth.pingone.us/02fb4743-189a-4bc7-9d6c-a919edfe6447/as")
assertTrue(copied.urlPath.revokeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/revoke")
assertTrue(copied.urlPath.tokenEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/token")
assertTrue(copied.urlPath.userinfoEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/userinfo")
assertTrue(copied.urlPath.endSessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/signoff")
assertTrue(copied.urlPath.sessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session")
assertTrue(copied.urlPath.authenticateEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate")
assertTrue(copied.oauth.oauthClientId == "AndroidTest")
assertTrue(copied.oauth.oauthRedirectUri == "org.forgerock.demo:/oauth2redirect")
assertTrue(copied.oauth.oauthScope == "openid profile email address phone")
}

@Test
fun testDiscoverEndpointWithURLProvided() {
runBlocking {
server.enqueue(
MockResponse()
fun testDiscoverEndpointWithURLNotProvided() = runTest {
server.enqueue(
MockResponse()
.setResponseCode(HttpURLConnection.HTTP_OK)
.addHeader("Content-Type", "application/json")
.setBody(getJson("/discovery.json")))

val option1 = FROptionsBuilder.build {
server {
url = "https://auth.pingone.us/02fb4743-189a-4bc7-9d6c-a919edfe6447/as"
}
urlPath {
authenticateEndpoint = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate"
sessionEndpoint = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session"
}
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
}
}
val copied = option1.discover(url)
assertTrue(copied.urlPath.authorizeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authorize")
assertTrue(copied.server.url == "https://auth.pingone.us/02fb4743-189a-4bc7-9d6c-a919edfe6447/as")
assertTrue(copied.urlPath.revokeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/revoke")
assertTrue(copied.urlPath.tokenEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/token")
assertTrue(copied.urlPath.userinfoEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/userinfo")
assertTrue(copied.urlPath.endSessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/signoff")
assertTrue(copied.urlPath.sessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session")
assertTrue(copied.urlPath.authenticateEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate")
assertTrue(copied.oauth.oauthClientId == "AndroidTest")
assertTrue(copied.oauth.oauthRedirectUri == "org.forgerock.demo:/oauth2redirect")
assertTrue(copied.oauth.oauthScope == "openid profile email address phone")
val option1 = FROptionsBuilder.build {
urlPath {
authenticateEndpoint =
"https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate"
sessionEndpoint =
"https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session"
}
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
}
}
val copied = option1.discover(url)
assertTrue(copied.urlPath.authorizeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authorize")
assertTrue(copied.server.url == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as")
assertTrue(copied.urlPath.revokeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/revoke")
assertTrue(copied.urlPath.tokenEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/token")
assertTrue(copied.urlPath.userinfoEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/userinfo")
assertTrue(copied.urlPath.endSessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/signoff")
assertTrue(copied.urlPath.sessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session")
assertTrue(copied.urlPath.authenticateEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate")
assertTrue(copied.oauth.oauthClientId == "AndroidTest")
assertTrue(copied.oauth.oauthRedirectUri == "org.forgerock.demo:/oauth2redirect")
assertTrue(copied.oauth.oauthScope == "openid profile email address phone")
}

@Test
fun testDiscoverEndpointWithURLNotProvided() {
runBlocking {
server.enqueue(
MockResponse()
.setResponseCode(HttpURLConnection.HTTP_OK)
.addHeader("Content-Type", "application/json")
.setBody(getJson("/discovery.json")))

val option1 = FROptionsBuilder.build {
urlPath {
authenticateEndpoint = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate"
sessionEndpoint = "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session"
}
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
}
}
val copied = option1.discover(url)
assertTrue(copied.urlPath.authorizeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authorize")
assertTrue(copied.server.url == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as")
assertTrue(copied.urlPath.revokeEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/revoke")
assertTrue(copied.urlPath.tokenEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/token")
assertTrue(copied.urlPath.userinfoEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/userinfo")
assertTrue(copied.urlPath.endSessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/signoff")
assertTrue(copied.urlPath.sessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/session")
assertTrue(copied.urlPath.authenticateEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/authenticate")
assertTrue(copied.oauth.oauthClientId == "AndroidTest")
assertTrue(copied.oauth.oauthRedirectUri == "org.forgerock.demo:/oauth2redirect")
assertTrue(copied.oauth.oauthScope == "openid profile email address phone")
fun testDiscoverEndpointWithPingEndIdpSession() = runTest {
//The discovery endpoint returns with ping_end_idp_session_endpoint
server.enqueue(
MockResponse()
.setResponseCode(HttpURLConnection.HTTP_OK)
.addHeader("Content-Type", "application/json")
.setBody(getJson("/discoveryWithPingEndIdp.json")))

//Since config without oauthSignOutRedirectUri, use ping_end_idp_session_endpoint instead of end_session_endpoint if exists
val option1 = FROptionsBuilder.build {
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
}
}
val copied = option1.discover(url)
//The ping_end_idp_session_endpoint is used instead of end_session_endpoint
assertTrue(copied.urlPath.endSessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/idp/signoff")
}

@Test
fun testDiscoverEndpointWithPingEndIdpSessionSignOutRedirect() = runTest {
//The discovery endpoint returns with ping_end_idp_session_endpoint
server.enqueue(
MockResponse()
.setResponseCode(HttpURLConnection.HTTP_OK)
.addHeader("Content-Type", "application/json")
.setBody(getJson("/discoveryWithPingEndIdp.json")))

}
val option1 = FROptionsBuilder.build {
oauth {
oauthClientId = "AndroidTest"
oauthRedirectUri = "org.forgerock.demo:/oauth2redirect"
oauthScope = "openid profile email address phone"
//Since config with oauthSignOutRedirectUri, use end_session_endpoint instead of ping_end_idp_session_endpoint
oauthSignOutRedirectUri = "org.forgerock.demo:/oauth2redirect"
}
}
//The end_session_endpoint is used instead of ping_end_idp_session_endpoint
val copied = option1.discover(url)
assertTrue(copied.urlPath.endSessionEndpoint == "https://auth.pingone.ca/02fb4743-189a-4bc7-9d6c-a919edfe6447/as/signoff")
}
}
Loading

0 comments on commit 0cb37c3

Please sign in to comment.