Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/cooperlyt/keycloakphoneclient
keycloak phone client sample for android
https://github.com/cooperlyt/keycloakphoneclient
android client keycloak phone sms
Last synced: about 1 month ago
JSON representation
keycloak phone client sample for android
- Host: GitHub
- URL: https://github.com/cooperlyt/keycloakphoneclient
- Owner: cooperlyt
- Created: 2021-01-03T04:52:31.000Z (almost 4 years ago)
- Default Branch: master
- Last Pushed: 2024-02-13T21:31:30.000Z (11 months ago)
- Last Synced: 2024-10-21T05:09:04.750Z (2 months ago)
- Topics: android, client, keycloak, phone, sms
- Language: Java
- Homepage:
- Size: 147 KB
- Stars: 7
- Watchers: 2
- Forks: 6
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Keycloak phone client
+ login and get token from keycloak
+ add token to http header, access protected resource
+ auto refresh tokenif you need phone support for keycloak , try my
project: [keycloak-phone-provider](https://github.com/cooper-lyt/keycloak-phone-provider), this sample is base
on: [keycloak-phone-provider](https://github.com/cooper-lyt/keycloak-phone-provider). this sample is android client,
nothing stop you from implementing other java program.## Usage
copy keycloak-client to your project
**Initialize:**
```java
accessTokenHolder=new KeycloakTokenHolder.Builder()
.host("www.XXX.com") // root address: like "www.XXX.com"
.realms("realms") // realms: config in your keycloak
.clientId("clientId") // clientId: config in your keycloak
.clientSecret("XXXX-XXXX-XXXX-XXXX-XXXXXXXXXX") //clientSecret : clientId: config in your keycloak
.storage(new SharedPreferencesStorage(getSharedPreferences("keycloak-sample",Context.MODE_PRIVATE),ACCESS_TOKEN_STORE_KEY)) // storage: you can custom for cc.coopersoft.accesstoken.keycloak.KeycloakTokenStorage
.build();```
storage: you can custom storage from KeycloakTokenStorage, in my project ,i use tencent mmkv.
**Login:**
1. Required auth code
```java
KeycloakClient.accessTokenHolder().sendAuthenticationCode(binding.phoneNumber.getText().toString(),new CodeRequestCallback(){
@Override
public void onFailure(String phoneNumber,ErrorResult error){
binding.requestAuthCode.setEnabled(true);
runOnUiThread(()->toast("send code fail!"));
}@Override
public void onSuccess(String phoneNumber,Result result){
Looper.prepare();
new CountDownTimer(result.getExpiresIn()*1000,1000){
@Override
public void onTick(long millisUntilFinished){
binding.requestAuthCode.setEnabled(false);
binding.requestAuthCode.setText("send("+millisUntilFinished/1000+")");}
@Override
public void onFinish(){
binding.requestAuthCode.setEnabled(true);
binding.requestAuthCode.setText("resend");}
}.start();
Looper.loop();
}
});```
2. Required token
```java
KeycloakClient.accessTokenHolder().requireToken(PhoneCredential.builder().phone(phone).code(code).build(), new AuthenticationCallback() {@Override
public void onFailure(ErrorResult error) {
runOnUiThread(() -> toast("login fail!"));
loading(false);
}@Override
public void onSuccess() {
MainActivity.start(LoginActivity.this);
finish();
}
});
```**Access protected resource:**
```java
okHttpClientBuilder.addInterceptor(new AllAccessTokenInterceptor());
```about Interceptor:
+ AllAccessTokenInterceptor
+ WhiteAccessTokenInterceptor
+ you can custom Interceptor on AccessTokenInterceptorin my project, I use Retrofit implement a dynamic Interceptor annotation like this:
```java
@POST("pyramid/task/receive/{id}")
@RequireInterceptor({"accessToken"})
Observable receive(@Path("id") int id);```
**Token auth refresh:**
1. auto refresh on token is expires.
2. auto refresh on http required first return 401.
3. refreshToke is expires, return 401.**verification phone number:**
1. Required verification code
```java
KeycloakClient.accessTokenHolder().sendVerificationCode(binding.phoneNumber.getText().toString(), new CodeRequestCallback() {
@Override
public void onFailure(String phoneNumber, ErrorResult error) {
binding.requestAuthCode.setEnabled(true);
runOnUiThread(()-> toast("send code fail!"));
}@Override
public void onSuccess(String phoneNumber, Result result) {
Looper.prepare();
new CountDownTimer(result.getExpiresIn() * 1000, 1000) {
@Override
public void onTick(long millisUntilFinished) {
binding.requestAuthCode.setEnabled(false);
binding.requestAuthCode.setText("send(" + millisUntilFinished / 1000 + ")");}
@Override
public void onFinish() {
binding.requestAuthCode.setEnabled(true);
binding.requestAuthCode.setText("resend");}
}.start();
Looper.loop();
}
});
```2. valid code
```java
KeycloakClient.accessTokenHolder().verificationCode(binding.phoneNumber.getText().toString(), binding.authCode.getText().toString(), new VerificationCallback() {
@Override
public void onError(String phoneNumber, ErrorResult error) {
runOnUiThread(()-> toast("valid fail!" + error.getErrorDescription()));
}@Override
public void onFailure(String phoneNumber) {
runOnUiThread(()-> toast("valid fail!"));
}@Override
public void onSuccess(String phoneNumber) {
runOnUiThread(()-> toast("valid success"));
}
});
```**logout (clear token):**
```java
KeycloakClient.accessTokenHolder().clearToken();
```**other:**
look for PhoneSupportAccessTokenHolder.