-
Notifications
You must be signed in to change notification settings - Fork 3
/
providers.lisp
148 lines (136 loc) · 6.17 KB
/
providers.lisp
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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
(in-package :clath)
(defun check-for-error (parameters)
(when-let
((err (assoc :error parameters))
(err-msg (assoc :error--description parameters)))
(error
(format nil "OpenID Server Error: ~a, ~a" (cdr err) (cdr err-msg))))
parameters)
(defparameter *provider-info*
`(
:facebook
(:string "Facebook"
:auth-endpoint "https://www.facebook.com/v20.0/dialog/oauth"
:token-endpoint "https://graph.facebook.com/v20.0/oauth/access_token"
:userinfo-endpoint "https://graph.facebook.com/v2.3/me"
:auth-scope "public_profile"
:fontawesome-icon "facebook")
:google
(:string "Google"
:endpoints-url
"https://accounts.google.com/.well-known/openid-configuration"
:auth-scope "openid profile email"
:fontawesome-icon "google")
:github
(:string "Github"
:auth-endpoint "https://github.com/login/oauth/authorize"
:token-endpoint "https://github.com/login/oauth/access_token"
:userinfo-endpoint "https://api.github.com/user"
;;FIXME: email endpoint not implemented yet.
:email-endpoint "https://api.github.com/user/emails"
:auth-scope ""
:fontawesome-icon "github")
:reddit
(:string "Reddit"
:auth-endpoint "https://www.reddit.com/api/v1/authorize"
:token-endpoint "https://www.reddit.com/api/v1/access_token"
:userinfo-endpoint "https://oauth.reddit.com/api/v1/me"
:auth-scope "identity"
:fontawesome-icon "reddit")
:stackexchange
(:string "Stack Exchange"
:auth-endpoint "https://stackexchange.com/oauth"
:token-endpoint "https://stackexchange.com/oauth/access_token"
;:userinfo-endpoint "https://api.stackexchange.com/2.1/me"
:userinfo-endpoint
"https://api.stackexchange.com/2.1/me?site=stackoverflow"
:auth-scope ""
:fontawesome-icon "stack-exchange")
:twitter
(:string "X.com"
:use-north t
:auth-endpoint "https://api.twitter.com/oauth/authorize"
:access-endpoint "https://api.twitter.com/oauth/access_token"
:request-endpoint "https://api.twitter.com/oauth/request_token"
:userinfo-endpoint
"https://api.twitter.com/1.1/account/verify_credentials.json"
:auth-scope ""
:fontawesome-icon "x-twitter")
:linkedin
(:string
"LinkedIn"
:auth-endpoint "https://www.linkedin.com/uas/oauth2/authorization"
:token-endpoint "https://www.linkedin.com/uas/oauth2/accessToken"
:userinfo-endpoint "https://api.linkedin.com/v2/userinfo"
:jwks-uri "https://www.linkedin.com/oauth/openid/jwks"
:auth-scope "openid email"
:fontawesome-icon "linkedin")
:yahoo
(:string "Yahoo"
:auth-endpoint "https://api.login.yahoo.com/oauth2/request_auth"
:token-endpoint "https://api.login.yahoo.com/oauth2/get_token"
:userinfo-endpoint ""
:auth-scope "identity"
;;Yahoo at some point was failing due to a URL with the string "yahoo" in it. Hence this
;; field. Yahoo is still failing, so whatever...
:url-string "yh"
:fontawesome-icon "yahoo")))
(defparameter *provider-secrets* nil)
(defgeneric request-user-info (provider access-token))
(defmethod request-user-info ((provider t) access-token)
;;FIXME: Doesn't handle failure
(cl-json:decode-json-from-string
(drakma:http-request (getf (provider-info provider) :userinfo-endpoint)
;;Facebook might not like alt param?
:parameters `(("access_token" . ,access-token))
:user-agent (user-agent provider))))
(defmethod request-user-info ((provider (eql :google)) access-token)
(cl-json:decode-json-from-string
(drakma:http-request (getf (provider-info provider) :userinfo-endpoint)
;;Facebook might not like alt param?
:parameters `(("alt" . "json")
("access_token" . ,access-token))
:basic-authorization (basic-authorization provider)
:user-agent (user-agent provider))))
(defmethod request-user-info ((provider (eql :reddit)) access-token)
(cl-json:decode-json-from-string
(drakma:http-request (getf (provider-info provider) :userinfo-endpoint)
:parameters `(("access_token" . ,access-token))
:method :get
:additional-headers
`(("Authorization"
. ,(format nil "bearer ~a" access-token)))
:user-agent (user-agent provider))))
(defmethod request-user-info ((provider (eql :github)) access-token)
(cl-json:decode-json-from-string
(drakma:http-request (getf (provider-info provider) :userinfo-endpoint)
:method :get
:additional-headers
`(("Authorization"
. ,(format nil "token ~a" access-token)))
:user-agent (user-agent provider))))
(defmethod request-user-info ((provider (eql :linkedin)) access-token)
(cl-json:decode-json-from-string
(drakma:http-request (getf (provider-info provider) :userinfo-endpoint)
:parameters `(("oauth2_access_token" . ,access-token))
:method :get
:user-agent (user-agent provider))))
(defmethod request-user-info ((provider (eql :stackexchange)) access-token)
(cadr
(assoc
:items
(cl-json:decode-json-from-string
(drakma:http-request (getf (provider-info provider) :userinfo-endpoint)
:parameters `(("key" . ,(getf (provider-secrets provider)
:key))
("access_token" . ,access-token))
:basic-authorization (basic-authorization provider)
:decode-content t
:user-agent (user-agent provider))))))
;;For now we will hope that all OAuth1.0a providers will behave the same.
;;If not, this can be turned into generic-and-methods.
(defun request-user-info-north (provider north-client)
(cl-json:decode-json-from-string
(north:make-signed-request
north-client
(getf (provider-info provider) :userinfo-endpoint) :get)))