一个成熟的软件系统一般必须有一个可靠的身份认证与权限验证功能。这一块要自研快速实现还是需要花费挺多精力的,幸好开源领域目前已经有不错的解决方案,一般拿过来按项目的实际需求进行一些简单的定制基本就可以实现业务目标了。最近刚好在这方面进行了一些工作,这里将如何实现身份认证及权限验证分两篇博文大概梳理一下,这篇先讲身份认证。
什么是身份认证
身份验证(英语:Authentication)又称“验证”、“鉴权”,是指通过一定的手段,完成对用户身份的确认。 身份验证的目的是确认当前所声称为某种身份的用户,确实是所声称的用户。在日常生活中,身份验证并不罕见;比如,通过检查对方的证件,我们一般可以确信对方的身份。虽然日常生活中的这种确认对方身份的做法也属于广义的“身份验证”,但“身份验证”一词更多地被用在计算机、通信等领域。
以上是维基百科的解释,说白了就是用某种方式确保用户是某种身份,这种确保需要保证其它用户没那么容易伪装其身份。一般只有经过身份认证得到可靠的用户身份后,才能基于该身份进行后续的权限验证流程。
实现身份认证系统
一般来说业务系统会专注于业务逻辑的处理,而身份认证相关的功能会放入独立的身份认证系统进行开发维护。要纯自研地完成一套可靠、安全度高的身份认证系统还是比较花费精力的,幸好开源领域目前已经有不错的解决方案,一般拿过来按项目的实际需求进行一些简单的定制基本就可以实现业务目标了。我这里的示例使用dex实现了一个简单的用户认证系统。
运行示例
前提是已经搭建好了go语言的开发环境,并设置好了GOPATH。
然后按以下步骤运行本程序:
代码语言:javascript复制# 编译dexserver
$ make build-dexserver
# 编译dexclient
$ make build-dexclient
# 运行dexserver
$ make run-dexserver
# 运行dexclient
$ run-dexclient
然后用浏览器访问http://127.0.0.1:8080, 页面会自动跳转至dexserver
的登录页面,输入用户名admin@example.com
、密码password
之后,会跳回dexclient
的callback页面http://127.0.0.1:8080/callback
。
示例的技术细节
dexserver
这里使用的dexserver
是由官方代码直接编译得出的,没有修改任何代码。只不过使用了自定义的配置文件dexserver-config.yaml,这里分析一下这个配置文件。
# The base path of dex and the external name of the OpenID Connect service.
# This is the canonical URL that all clients MUST use to refer to dex. If a
# path is provided, dex's HTTP service will listen at a non-root URL.
issuer: http://127.0.0.1:5556/dex
# The storage configuration determines where dex stores its state. Supported
# options include SQL flavors and Kubernetes third party resources.
#
# See the storage document at Documentation/storage.md for further information.
storage:
type: sqlite3
config:
file: config/dex.db
# Configuration for the HTTP endpoints.
web:
http: 0.0.0.0:5556
# Uncomment for HTTPS options.
# https: 127.0.0.1:5554
# tlsCert: /etc/dex/tls.crt
# tlsKey: /etc/dex/tls.key
# Configuration for telemetry
telemetry:
http: 0.0.0.0:5558
# Uncomment this block to enable the gRPC API. This values MUST be different
# from the HTTP endpoints.
# grpc:
# addr: 127.0.0.1:5557
# tlsCert: examples/grpc-client/server.crt
# tlsKey: examples/grpc-client/server.key
# tlsClientCA: /etc/dex/client.crt
# Uncomment this block to enable configuration for the expiration time durations.
# expiry:
# signingKeys: "6h"
# idTokens: "24h"
# Options for controlling the logger.
# logger:
# level: "debug"
# format: "text" # can also be "json"
# Uncomment this block to control which response types dex supports. For example
# the following response types enable the implicit flow for web-only clients.
# Defaults to ["code"], the code flow.
# oauth2:
# responseTypes: ["code", "token", "id_token"]
oauth2:
skipApprovalScreen: true
# Instead of reading from an external storage, use this list of clients.
#
# If this option isn't chosen clients may be added through the gRPC API.
staticClients:
- id: demo-dexclient
redirectURIs:
- 'http://127.0.0.1:8080/callback'
name: 'Demo dex client'
secret: ZXhhbXBsZS1hcHAtc2VjcmV0
connectors: []
# - type: mockCallback
# id: mock
# name: Example
# - type: oidc
# id: google
# name: Google
# config:
# issuer: https://accounts.google.com
# # Connector config values starting with a "$" will read from the environment.
# clientID: $GOOGLE_CLIENT_ID
# clientSecret: $GOOGLE_CLIENT_SECRET
# redirectURI: http://127.0.0.1:5556/dex/callback
# hostedDomains:
# - $GOOGLE_HOSTED_DOMAIN
# Let dex keep a list of passwords which can be used to login to dex.
enablePasswordDB: true
# A static list of passwords to login the end user. By identifying here, dex
# won't look in its underlying storage for passwords.
#
# If this option isn't chosen users may be added through the gRPC API.
# staticPasswords:
# - email: "admin@example.com"
# # bcrypt hash of the string "password"
# hash: "$2a$10$2b2cU8CPhOTaGrs1HRQuAueS7JTT5ZHsHSzYiFPm1leZck7Mc8T4W"
# username: "admin"
# userID: "08a8684b-db88-4b73-90a9-3cd1661f5466"
web
段配置的是dexserver
的监听地址及HTTPS证书参数,issuer
配置的是外部会访问到的系统URL,这两者一般要对应地设置。
telemetry
段配置的是监控指标抓取地址,本例中dexserver
启动完毕后,可访问http://127.0.0.1:5558/metrics抓取到该dexserver
的监控指标。
storage
段配置的是dexserver
的存储设置。dexserver
在运行时跟踪refresh_token
、auth_code
、keys
、password
等的状态,因此需要将这些状态保存下来。dex提供了多种存储方案,如etcd
、CRDs
、SQLite3
、Postgres
、MySQL
、memory
,总有一款能满足需求。如果要其它需求,还可以参考现有Storage及文档扩展一个。我这里使用的是比较简单的SQLite3
Storage,提前插入了一条测试的用户数据:
sqlite3 config/dex.db
sqlite> insert info password values('admin@example.com', '$2a$10$2b2cU8CPhOTaGrs1HRQuAueS7JTT5ZHsHSzYiFPm1leZck7Mc8T4W', 'admin', '08a8684b-db88-4b73-90a9-3cd1661f5466');
sqlite> .quit
oauth2.skipApprovalScreen
这个选项我设置成了true
,这样就不会有提示用户同意的页面出现。
staticClients
段配置的是该dexserver
允许接入的dexclient
信息,这个要跟dexclient
那边的配置一致。
connectors
段我并没有配置任何Connector
。Connector
是dex
中一项重要特性,其可以将dex
这个身份认证系统与其它身份认证系统串联起来。dex
目前自带的Connector
有LDAP
、GitHub
、SAML 2.0
、GitLab
、OpenID Connect
、LinkedIn
、Microsoft
、AuthProxy
、Bitbucket Cloud
,基本上满足绝大部分需求,如果要扩展,参考某个现成的Connector实现即可。我这个示例里因为直接使用保存在DB里的帐户密码信息,因此只需要配置enablePasswordDB
为true
,就会自动使用上passwordDB
这个Connector
,passwordDB
的实现代码见这里。
最近由于登录页面是由dexserver
提供了,这里还将dex
自带的登录页面web端资源带上了,具体的项目中根据场景对页面进行一些定制就可以了。
dexclient
dexclient
就很简单了,就两个go文件,重点是cmd/dexclient/main.go
。
首先是根据一系列参数构造出oidc.Provider
及oidc.IDTokenVerifier
,这个后面获取认证系统的跳转地址、获取id_token
、校验id_token
都会用到:
...
a.provider = provider
a.verifier = provider.Verifier(&oidc.Config{ClientID: a.clientID})
然后声明处理三个请求地址,并启动Web Server:
代码语言:javascript复制 http.HandleFunc("/", a.handleIndex)
http.HandleFunc("/login", a.handleLogin)
http.HandleFunc(u.Path, a.handleCallback)
switch listenURL.Scheme {
case "http":
log.Printf("listening on %s", listen)
return http.ListenAndServe(listenURL.Host, nil)
case "https":
log.Printf("listening on %s", listen)
return http.ListenAndServeTLS(listenURL.Host, tlsCert, tlsKey, nil)
default:
return fmt.Errorf("listen address %q is not using http or https", listen)
}
很明显handleIndex
就是WEB应用的主页,这里一般逻辑应该是检查用户的登录身份信息是否合法,如果不合法则跳至dexserver
的登录页面。
var indexTmpl = template.Must(template.New("index.html").Parse(`<html>
<!-- TODO Redirect to login page if not logged -->
<body>
<form action="/login" method="post">
<p>
Authenticate for:<input type="text" name="cross_client" placeholder="list of client-ids">
</p>
<p>
Extra scopes:<input type="text" name="extra_scopes" placeholder="list of scopes">
</p>
<p>
Connector ID:<input type="text" name="connector_id" placeholder="connector id">
</p>
<p>
Request offline access:<input type="checkbox" name="offline_access" value="yes" checked>
</p>
<input type="submit" value="Login" id="submitBtn">
</form>
</body>
<script type="text/javascript">
<!-- Redirect to login page -->
document.getElementById("submitBtn").click();
</script>
</html>`))
handleLogin
根据浏览器发来的cross_client
、extra_scopes
、connector_id
、offline_access
参数构造出登录页跳转地址,并提示浏览器跳至该地址:
...
if r.FormValue("offline_access") != "yes" {
authCodeURL = a.oauth2Config(scopes).AuthCodeURL(exampleAppState)
} else if a.offlineAsScope {
scopes = append(scopes, "offline_access")
authCodeURL = a.oauth2Config(scopes).AuthCodeURL(exampleAppState)
} else {
authCodeURL = a.oauth2Config(scopes).AuthCodeURL(exampleAppState, oauth2.AccessTypeOffline)
}
if connectorID != "" {
authCodeURL = authCodeURL "&connector_id=" connectorID
}
http.Redirect(w, r, authCodeURL, http.StatusSeeOther)
handleCallback
处理登录成功后的回调请求,其根据回调请求中的code
参数,调用dexserver
的相关接口换取包含用户身份信息的Token
:
code := r.FormValue("code")
if code == "" {
http.Error(w, fmt.Sprintf("no code in request: %q", r.Form), http.StatusBadRequest)
return
}
if state := r.FormValue("state"); state != exampleAppState {
http.Error(w, fmt.Sprintf("expected state %q got %q", exampleAppState, state), http.StatusBadRequest)
return
}
token, err = oauth2Config.Exchange(ctx, code)
一般来说,会将该Token
中的id_token
进行适当的编码发回到浏览器中保存(以Cookie或WebStorage等方式),这样浏览器中就保存了用户的身份信息。
安全起见,dexserver
签发的id_token
有效期通常不会太长,这就需要dexclient
凭借Token
中的refresh_token
隔段时间重新换取新的Token
,并通过某种机制将新Token
中的id_token
重新发回浏览器端保存。以refresh_token
重新换取新的Token
的代码实现如下:
t := &oauth2.Token{
RefreshToken: refresh,
Expiry: time.Now().Add(-time.Hour),
}
token, err = oauth2Config.TokenSource(ctx, t).Token()
项目定制扩展
上面的示例基本说明得比较清楚了,在真实的生产项目需要进行必要的定制以满足项目需求,生产项目的代码必须保密,这里只简要说明下关键点:
- 生产系统,无论
dexserver
还是dexclient
都必须配置HTTPS证书,通过HTTPS协议访问站点。 dexserver
会根据项目情况配置一个更合适的Storage
,用以安全可靠地保存refresh_token
、auth_code
、keys
、password
等的状态。要考虑这个Storage
实现方案的性能、稳定性、高可用性等多个因素。- 一般还需要对保存进
Storage
的帐户身份信息进行维护管理,不可能每次都命令行增删帐户身份信息。 - 示例中
dexserver
直接使用的passwordDB
这个Connector
,实际场景可能会扩展passwordDB
,比如需要往返回的claims
里植入Groups
等其它信息。 - 示例中
dexserver
只使用了passwordDB
这个Connector
,实际场景为了接入其它身份认证系统,会配置一些Connect
以对接其它身份认证系统。 - 示例中
dexserver
只使用了dex
官方自带的登录页模板,这个放在实际项目中还是太简陋了,一般需要一些定制美化。 - 示例中
dexclient
并没有在浏览器中完成用户身份信息验证,以决定是否跳转登录页面,这个在实际项目中肯定是要做的。 - 示例中
dexclient
并没有根据用户的选择跳转至指定的Connector
登录页,对于支持多种认证方式的业务系统来说,这个也是必须的。 - 示例中
dexclient
并没有实现将Token
中的id_token
进行适当的编码发回到浏览器中保存的逻辑,这个一般也是必须的。 - 示例中
dexclient
并没有凭借Token
中的refresh_token
隔段时间重新换取新的Token
,并通过某种机制将新Token
中的id_token
重新发回浏览器端保存,这个一般也是必须的。
总之这个示例要最终在生产项目中落地,还是要不少工作要处理的,不过要了解其核心工作原理,看这个示例也就差不多了。
总结
总体来说,采用dex
实现一个身份认证系统相比纯自研,还是简单了不少。一旦掌握其原理,后续实现会很方便,其核心代码值得学习一下。