aboutsummaryrefslogtreecommitdiffstats
path: root/notes/auth.md
blob: d5e4dbd414722e8290721e06e1574ac9be63e108 (plain)
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
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183

For users: use openid connect (oauth2) to sign up and login to web app. From
web app, can create (and disable?) API tokens

For impl: fatcat-web has private key to create tokens. tokens used both in
cookies and as API keys. tokens are macaroons (?). fatcatd only verifies
tokens. optionally, some redis or other fast shared store to verify that tokens
haven't been revoked.

Could use portier with openid connect as an email-based option. Otherwise,
orcid, github, google.

---------

Use macaroons!

editor/user table has a "auth_epoch" timestamp; only macaroons generated
after this timestamp are valid. revocation is done by incrementing this
timestamp ("touch").

Rust CLI tool for managing users:
- create editor

Special users/editor that can create editor accounts via API; eg, one for
fatcat-web.

Associate one oauth2 id per domain per editor/user.

Users come to fatcat-web and do oauth2 to login or create an account. All
oauth2 internal to fatcat-web. If successful, fatcat-web does an
(authenticated) lookup to API for that identifier. If found, requests a
new macaroon to use as a cookie for auth. All future requests pass this
cookie through as bearer auth. fatcat-web remains stateless! macaroon
contains username (for display); no lookup-per page. Need to logout/login for
this to update?

Later, can do a "add additional account" feature.

Backend:
- oauth2 account table, foreign key to editor table
    => this is the only private table
- auth_epoch timestamp column on editor table
- lock editor by setting auth_epoch to deep future

Deploy process:
- auto-create root (admin), import-bootstrap (admin,bot), and demo-user
  editors, with fixed editor_id and "early" auth_epoch, as part of SQL. save
  tokens in env files, on laptop and QA instance.
- on live QA instance, revoke all keys when live (?)

TODO: privacy policy

fatcat API doesn't *require* auth, but if auth is provided, it will check
macaroon, and validate against editor table's timestamp.

support oauth2 against:
- orcid
- git.archive.org
- github
? google

Macaroon details:
- worth looking at "bakery" projects (python and golang) for example of how to
  actually implement macaroon authentication/authorization
- location is fatcat.wiki (and/or qa.fatcat.wiki, or test or localhost or test.fatcat.wiki?)
- identifier is a UUID in upper-case string format
- will need some on-disk key storage thing?
    => how to generate new keys? which one should be used, most recent?
       conception of revoking keys? simple JSON/TOML, or LMDB?
- call them "authentication tokens"?
- params/constraints
    - editor_id: always, fcid format
    - created: always, some date format (seconds/iso)
    - expires: optional, same date format

It's a huge simplification to have webface generate macaroons as well, using a
root key. webface doesn't need multiple keys because it only creates, doesn't
verify.

Code structure:
- auth service/struct is generated at startup; reads environment and on-disk keys
- verify helper does the thing
- some sort of auth/edit context

Roles?
- public: unauthenticated
- editor: any authenticated, active account
- bot
- admin

Caveats:
- general model is that macaroon is omnipotent and passes all verification,
  unless caveats are added. eg, adding verification checks doesn't constrain
  auth, only the caveats constrain auth; verification check *allow* additional
  auth. each caveat only needs to be allowed by one verifiation.
- can (and should?) add as many caveat checkers/constrants in code as possible

http://evancordell.com/2015/09/27/macaroons-101-contextual-confinement.html

-------

## Schema/API Notes

GET /auth/oidc
=> params: provider, sub, iss
=> returns {editor, token} or not found
=> admin auth required

POST /auth/oidc
=> params: editor_id, provider, sub, iss
=> returns {editor, token}
=> admin auth required

POST /editor
=> admin auth required

flow is to have single login/signup OIDC flow. If need to create an account,
bounce to special page for that and store ISS/SUB in (signed/secure) session
temporarily.

This doesn't feel great. Could instead randomly generate a username, and
provide mechanism to update. That's better!

PUT /editor/{editor_id}
=> only allow username updates, and only by admin or logged-in user

schema:
`auth_oidc`
    => id (BIGINT), editor_id, provider, oidc_iss, oidc_sub
    => created (auto-timestamp)
    => UNIQ index on (editor_id, provider)
    => UNIQ index on (provider, remote_sub, remote_iss)
    => all are NOT NULL

## Webface Notes

Want to use "OpenID Connect" (OIDC), which is basically a subset/convention of
OAuth 2.0 for authenticaiton ("log in as"), without granting API priviliges.

Want to support multiple identity providers, eg:
- orcid.org
    => Basic OpenID Provider; implicit token
- git.archive.org
- gitlab.org
    => https://docs.gitlab.com/ee/integration/openid_connect_provider.html
- google.com

Currently, looks like github.com doesn't support OIDC; they are the only
provider i'm interested in that does not.

authlib/loginpass are tempting to use as they support a bunch of providers
out-of-the-box... but not orcid.

Alternatively, could use any number of "proxies"/thingies to aggregate auth:
- https://www.keycloak.org/about.html
- https://portier.github.io/
- https://github.com/dexidp/dex

Possible flask integrations:
=> https://flask-oidc.readthedocs.io/en/latest/
=> https://github.com/zamzterz/Flask-pyoidc

Background:
=> https://blog.runscope.com/posts/understanding-oauth-2-and-openid-connect
=> https://latacora.micro.blog/2018/06/12/a-childs-garden.html

Future work:
=> multiple logins, and/or merging accounts


"Fatcat is an open, editable database of bibliographic metadata. You can
sign-up and login using orcid.org; this option is used for identity and
authentication only. Fatcat does not currently make changes to any data on
orcid.org, which you can verify from the permissions requested."

    https://fatcat.wiki/auth/oidc_redirect
    https://qa.fatcat.wiki/auth/oidc_redirect

PLAN:
- have a mode/mechanism for login-by-token; mostly for testing
- for now, use loginpass OAuth/OIDC for login/signup. upstream ORCID support or
  hack that in somehow when desired
- auto-create a username based on oauth, then allow changes