forked from CERN-CERT/pam_2fa
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathyk.c
86 lines (73 loc) · 2.69 KB
/
yk.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "pam_2fa.h"
#include "curl.h"
int yk_auth_func(pam_handle_t * pamh, module_config * cfg, const char* username, const char *otp);
const auth_mod yk_auth = {
.do_auth = &yk_auth_func,
.name = "Yubikey",
.prompt = "Yubikey: ",
.otp_len = YK_OTP_LEN,
};
static int valid_otp(pam_handle_t * pamh, module_config * cfg, const char *otp)
{
unsigned int i = 0;
for (i = 0; otp[i]; ++i) {
if (!isalpha(otp[i])) {
DBG_C(pamh, cfg, "INCORRRECT code from user!");
return 0;
}
}
if (i != YK_OTP_LEN) {
DBG_C(pamh, cfg, "INCORRRECT code from user!");
return 0;
}
return 1;
}
/* There is no escape issue as both the username and key can't contain '"' */
static const char * const yk_association_request = "{\"username\": \"%s\", \"yubicode\": \"%s\"}";
static const char * const yk_association_ok = "true";
int yk_auth_func(pam_handle_t * pamh, module_config * cfg, const char* username, const char *otp)
{
struct pam_curl_state* state;
struct curl_response* resp;
char * payload;
int return_value = PAM_AUTH_ERR;
DBG_C(pamh, cfg, "Yubikey = %s", otp);
if (!valid_otp(pamh, cfg, otp)) {
USER_ERR_C(pamh, cfg, "Invalid OTP provided by user");
return PAM_AUTH_ERR;
}
state = pam_curl_init(pamh, cfg);
if (state == NULL) {
/* Errors already logged in pam_curl_init */
return 0;
}
resp = (struct curl_response*) calloc(1, sizeof(struct curl_response));
if (resp == NULL) {
ERR_C(pamh, cfg, "Yubikey: unable to allocate buffer for curl response");
goto clean_state;
}
if (asprintf(&payload, yk_association_request, username, otp) < 0) {
ERR_C(pamh, cfg, "Yubikey: unable to allocate buffer for curl payload");
goto clean_resp;
}
/* add_header and set_option already log errors */
PAM_CURL_DO_OR_GOTO(state, add_header, clean_payload, "Content-Type: text/json");
PAM_CURL_DO_OR_GOTO(state, set_option, clean_payload, CURLOPT_FAILONERROR, 1);
PAM_CURL_DO_OR_GOTO(state, set_option, clean_payload, CURLOPT_WRITEFUNCTION, &curl_callback_copy);
PAM_CURL_DO_OR_GOTO(state, set_option, clean_payload, CURLOPT_WRITEDATA, resp);
PAM_CURL_DO_OR_GOTO(state, set_option, clean_payload, CURLOPT_URL, cfg->yk_uri);
PAM_CURL_DO_OR_GOTO(state, set_option, clean_payload, CURLOPT_POSTFIELDS, payload);
if (pam_curl_perform(state) == CURLE_OK && strcmp(yk_association_ok, resp->buffer) == 0) {
return_value = PAM_SUCCESS;
}
clean_payload:
free(payload);
clean_resp:
free(resp);
clean_state:
pam_curl_cleanup(state);
return return_value;
}