keycloak中的client-scope允许你为每个客户端分配scope,而scope就是授权范围,它直接影响了token中的内容,及userinfo端点可以获取到的用户信息,这块我们可以通过自定义scope/mapper,来实现粒度的控制,并且这个mapper可以控制添加到token,或者添加到userinfo端点,这两块配置也是独立的,下面我们通过一个登录IP地址的mapper,来实现将登录ip添加到token和userinfo端点。
添加Mapper对象
public class ExtensionLoginIpMapper
extends AbstractOIDCProtocolMapper
implements OIDCAccessTokenMapper, OIDCIDTokenMapper, UserInfoTokenMapper {
public static final String CONFIG_NAME = "ExtensionLoginIp";//配置里的名称
public static final String PROVIDER_ID = "oidc-extension-login-ip-mapper";
private static final List<ProviderConfigProperty> configProperties = new ArrayList<ProviderConfigProperty>();
private static final String LOGIN_IP = "loginIp";
static {
configProperties.add(createConfigProperty(CONFIG_NAME, "Token申请名", "在jwt中的属性名称,默认loginIp"));
OIDCAttributeMapperHelper.addIncludeInTokensConfig(configProperties, ExtensionLoginIpMapper.class);
}
protected static ProviderConfigProperty createConfigProperty(String claimName, String label, String help) {
ProviderConfigProperty property = new ProviderConfigProperty();
property.setName(claimName);
property.setLabel(label);
property.setHelpText(help);
property.setType(ProviderConfigProperty.STRING_TYPE);
return property;
}
@Override
protected void setClaim(IDToken token, ProtocolMapperModel mappingModel, UserSessionModel userSession,
KeycloakSession keycloakSession, ClientSessionContext clientSessionCtx) {
try {
String key = LOGIN_IP;
if (mappingModel.getConfig().containsKey(CONFIG_NAME)) {
key = mappingModel.getConfig().get(CONFIG_NAME);
}
if (userSession.getNotes().containsKey(LOGIN_IP)) {
String val = userSession.getNote(LOGIN_IP);
token.setOtherClaims(key, val);
}
} catch (Exception e) {
e.printStackTrace();
}
}
public List<ProviderConfigProperty> getConfigProperties() {
return configProperties;
}
@Override
public String getId() {
return PROVIDER_ID;
}
@Override
public String getDisplayType() {
return CONFIG_NAME;
}
@Override
public String getDisplayCategory() {
return TOKEN_MAPPER_CATEGORY;
}
@Override
public String getHelpText() {
return "Maps Extension Login Ip Address.";
}
}
将Mapper添加到Client Scope
- 添加 client scope
data:image/s3,"s3://crabby-images/f4356/f43567d67a98aae15afaaa97344961b50a8888f2" alt=""
- 在client scope中添加mapper
data:image/s3,"s3://crabby-images/6faca/6faca97f2213025f7ab44f4039497a5df1a2f2d7" alt=""
设置access_token可见和userinfo可见
- Add to ID token
- Add to access token
- Add to access token
data:image/s3,"s3://crabby-images/1779f/1779f86f81aa61be08e1d0bfa1eaaf417fb37f37" alt=""
为客户端指定scope
- 这对于根据客户端来控制token和userinfo端点是非常必要的功能
- 这是oauth2授权的重要组成部分
data:image/s3,"s3://crabby-images/7a4c9/7a4c92e8e71acf718f368e267d03c511e3a9346c" alt=""
通过oauth2中的密码认证时的注意点
- 客户端不能是
同意必选
的,这种客户端需要通过浏览器认证,由用户自己确认
它公开的信息
data:image/s3,"s3://crabby-images/8515e/8515e08dd5ac53f7acf2b5fdac41c4b928d9d5a6" alt=""
data:image/s3,"s3://crabby-images/28ab1/28ab18f1d4c33eb70780852da79568f9928f72b9" alt=""
通过token获取用户信息
- userinfo端点:/auth/realms/{realms}/protocol/openid-connect/userinfo
- 获取到时的用户信息中的字段,是通过scope来控制的
data:image/s3,"s3://crabby-images/f3bb5/f3bb5dd14e1be89668219ebea77dcecc780b70c6" alt=""