comparison mod_auth_oauth_external/mod_auth_oauth_external.lua @ 5344:0a6d2b79a8bf

mod_auth_oauth_external: Authenticate against an OAuth 2 provider But suddenly unsure whether this constitutes an OAuth "client" or something else? Resource server maybe?
author Kim Alvefur <zash@zash.se>
date Thu, 16 Mar 2023 12:45:22 +0100
parents
children 3390bb2f9f6c
comparison
equal deleted inserted replaced
5343:5c1c70e52635 5344:0a6d2b79a8bf
1 local http = require "net.http";
2 local async = require "util.async";
3 local json = require "util.json";
4 local sasl = require "util.sasl";
5
6 -- TODO -- local issuer_identity = module:get_option_string("oauth_external_issuer");
7 local oidc_discovery_url = module:get_option_string("oauth_external_discovery_url")
8 local validation_endpoint = module:get_option_string("oauth_external_validation_endpoint");
9
10 local username_field = module:get_option_string("oauth_external_username_field", "preferred_username");
11
12 -- XXX Hold up, does whatever done here even need any of these things? Are we
13 -- the OAuth client? Is the XMPP client the OAuth client? What are we???
14 -- TODO -- local client_id = module:get_option_string("oauth_external_client_id");
15 -- TODO -- local client_secret = module:get_option_string("oauth_external_client_secret");
16
17 --[[ More or less required endpoints
18 digraph "oauth endpoints" {
19 issuer -> discovery -> { registration validation }
20 registration -> { client_id client_secret }
21 { client_id client_secret validation } -> required
22 }
23 --]]
24
25 local host = module.host;
26 local provider = {};
27
28 function provider.get_sasl_handler()
29 local profile = {};
30 profile.http_client = http.default; -- TODO configurable
31 local extra = { oidc_discovery_url = oidc_discovery_url };
32 function profile:oauthbearer(token)
33 if token == "" then
34 return false, nil, extra;
35 end
36
37 local ret, err = async.wait_for(self.profile.http_client:request(validation_endpoint,
38 { headers = { ["Authorization"] = "Bearer " .. token; ["Accept"] = "application/json" } }));
39 if err then
40 return false, nil, extra;
41 end
42 local response = ret and json.decode(ret.body);
43 if not (ret.code >= 200 and ret.code < 300) then
44 return false, nil, response or extra;
45 end
46 if type(response) ~= "table" or type(response[username_field]) ~= "string" then
47 return false, nil, nil;
48 end
49
50 return response[username_field], true, response;
51 end
52 return sasl.new(host, profile);
53 end
54
55 module:provides("auth", provider);