forked from StatisMike/shiny.reglog
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathREADME.Rmd
415 lines (306 loc) · 14.2 KB
/
README.Rmd
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
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
---
output: github_document
---
<!-- README.md is generated from README.Rmd. Please edit that file -->
```{r, include = FALSE}
knitr::opts_chunk$set(
collapse = TRUE,
comment = "#>",
fig.path = "man/figures/README-",
out.width = "100%"
)
```
# shiny.reglog
<!-- badges: start -->
[![Lifecycle: stable](https://img.shields.io/badge/lifecycle-stable-green.svg)](https://lifecycle.r-lib.org/articles/stages.html)
[![CRAN status](https://www.r-pkg.org/badges/version/shiny.reglog)](https://CRAN.R-project.org/package=shiny.reglog)
[![CRAN checks](https://cranchecks.info/badges/worst/shiny.reglog)](https://cranchecks.info/pkgs/shiny.reglog)
[![](http://cranlogs.r-pkg.org/badges/last-month/shiny.reglog?color=green)](https://cran.r-project.org/package=shiny.reglog)
[![](http://cranlogs.r-pkg.org/badges/grand-total/shiny.reglog?color=blue)](https://cran.r-project.org/package=shiny.reglog)
[![Codecov test coverage](https://codecov.io/gh/StatisMike/shiny.reglog/branch/master/graph/badge.svg)](https://app.codecov.io/gh/StatisMike/shiny.reglog?branch=master)
[![R-CMD-check](https://github.com/StatisMike/shiny.reglog/workflows/R-CMD-check/badge.svg)](https://github.com/StatisMike/shiny.reglog/actions)
<!-- badges: end -->
## 1. Introduction
The user authentication in Shiny applications can be very useful. Mainly, user can
login to read and write some results of their session into relational database.
On the other hand, it may be handy for your App to allow access of unregistered
users. If you need to secure your ShinyApp, there are better alternatives
([shinymanager](https://github.com/datastorm-open/shinymanager){target="_blank"} or
[shinyauthr](https://github.com/PaulC91/shinyauthr){target="_blank"})
This package contains modules to use in your Shiny application allowing you to
automatically insert boxes for login, register, credentials edit and password
reset and procedures.
*shiny.reglog* supports as data containers either databases accessed with
`RSQLite`, `RMariaDB`, `RMySQL` and `RPostgreSQL` drivers or googlesheets-based
database (accessed by `googlesheets4` package).
It is highly recommended to use one of the `DBI`-supported databases, though. It
is much more optimized and secure, as the database is never loaded as a whole
into the memory, but queried as needed. *googlesheets* database is much easier to
set-up, but it shouldn't be used when you are expecting big userbase.
Registration, credentials edit and password reset procedures programmatically
send email to the user of your ShinyApp - to welcome them, inform about change
of their user ID and/or email and to give them a reset code to reset their
password. *shiny.reglog* supports two methods of email sending: via `emayili`
or `gmailr` packages. Both of them have their pros and cons, depending on your
accesses: `emayili` allows for usage of many SMTP servers, while `gmailr` allowing
using *gmail* messaging via Google REST API.
The `emayili` is recommended for most applications. `gmailr` can be useful
if you already have application registered and authorized with mail sending
scope.
>Currently the package is after major change in its code - basically full rewrite
to allow more security, usage of more databases and more customization. Past
functions are still available in current version, but will
generate deprecation warnings.
## 2. Additional information
Basic information about *shiny.reglog* is contained within this document. There
are some more resources to learn about its usage:
- You can access demonstration shinyApp with showcase mode on
[shinyapps.io](https://statismik.shinyapps.io/shinyreglog_demo/){target="_blank"}
- You can run interactive demo in your own RStudio by using `RegLogDemo()`
function. It will use mocked *mailConnector* by default or *RegLogEmayiliConnector*
if you provide it arguments required by `emayili` backend.
- For more information read vignettes.
## 3. Basic structure
There are three main objects that are to be used when implementing *RegLog* system
for login and registration in your ShinyApp. All of them need to be defined
in the *server* code.
- *dbConnector*: one of `RegLogDBIConnector` or `RegLogGsheetConnector`. It will handle
all writes, edits and inputs to the users database.
- *mailConnector*: one of `RegLogEmayiliConnector` or `RegLogGmailrConnector`. It
will handle all automatical emailing to the user emails.
- *RegLogServer*: The main module producing and controlling all of UI and
reactivity of the login and registration system. It uses *dbConnector* and
*mailConnector* for actions external of R.
## 4. Installation
You can install this version of shiny.reglog from GitHub with:
```{r installation, eval=FALSE}
# install last stable release from CRAN
install.packages("shiny.reglog")
## or development version from GitHub
install.packages("devtools")
devtools::install.github("StatisMike/shiny.reglog")
```
## 5. Setting up *dbConnector*
You need to create *dbConnector* object to be used by the *RegLogServer* to
write and read user data from the database.
To set-up the database for *RegLog* system, you can use helper functions
included in this package. They are tested and guarantee compatible structure
of the data.
### 5.1 Googlesheet database method (*RegLogGsheetConnector*)
1. Create googlesheet file on your googledrive to support database. You can use
`gsheet_tables_create()` function, which by default creates empty spreadsheets
configured correctly.
```{r creating gsheet, eval = F}
# create googlesheet and gather its id for later usage
# you can also specify optional 'name' argument for custom gsheet name
gsheet_id <- gsheet_tables_create()
# save you gsheet_id - you need to provide it later to your dbConnector
```
If you wish to import some existing credentials, you can do it by giving the
`data.frame` object to the `user_data` argument:
```{r, eval = F}
# get some credentials
credentials <- data.frame(
username = "ShinyReglogTest",
password = "VeryHardPassword",
email = "shinyreglog@test"
)
# create gsheet database with some credentials
gsheet_id <- gsheet_tables_create(
user_data = credentials,
# as the password was not hashed with `script` before,
# it need to be hashed now
hash_passwords = T)
```
2. Configure googlesheets4 package to use out-of-band (non-interactive) auth. For more information about it visit [googlesheets4 documentation](https://googlesheets4.tidyverse.org/).
3. In the `server` part of your ShinyApp define `RegLogGsheetConnector` to provide it
afterwards to the `RegLogServer` object
```{r, eval = F}
server <- function(input, output, session) {
dbConnector <- RegLogGsheetConnector$new(
gsheet_ss = gsheet_id)
}
```
### 5.2 `DBI` compatible SQL database (*RegLogDBIConnector*)
*RegLog system* out of the box supports *SQLite*, *MySQL*, *MariaDB* and *PostgreSQL*
databases. You can use `DBI_tables_create` function, which by default creates empty
tables configured correctly.
```{r creating SQLite, eval = F}
# create a connection to the database. Remember to use user with CREATE TABLE
# scope enabled when useing MySQL, MariaDB or PostgreSQL connection
conn <- DBI::dbConnect(
RSQLite::SQLite(),
dbname = "reglog_db.sqlite"
)
# using this connection create the tables.
DBI_tables_create(conn = conn)
# disconnect from the database after creation
DBI::dbDisconnect(conn)
```
If you wish to import some credentials, you can do it by providing the `data.frame`
object to the `user_data` argument:
```{r, eval = F}
# get some credentials
credentials <- data.frame(
username = "ShinyReglogTest",
password = "VeryHardPassword",
email = "shinyreglog@test")
conn <- DBI::dbConnect(
RSQLite::SQLite(),
dbname = "reglog_db.sqlite"
)
# create database using the connection
DBI_tables_create(
conn = conn,
user_data = credentials,
# as the password was not hashed with `script` before,
# it need to be hashed now
hash_passwords = T)
DBI::dbDisconnect(conn)
```
2. In the *server* part of your ShinyApp define `RegLogDBIConnector` to provide it
afterwards to the `RegLogServer` object.
```{r, eval = F}
server <- function(input, output, session) {
dbConnector <- RegLogDBIConnector$new(
driver = RSQLite::SQLite(),
dbname = "reglog_db.sqlite")
}
```
### 5.3. `MongoDB`-based connector (*RegLogMongoConnector*)
*MongoDB* is very popular NoSQL database with pretty low entry-point. Connector
is still experimental.
Setup is analogous to other database connectors. For more informations, refer
to documentation: `?mongo_tables_create()`, `?RegLogMongoConnector`
## 6. Setting up mail connectors
You need to create *mailConnector* object to be used by the *RegLogServer* to
write and read user data from the database. There are two classes defined to use
`emayili` or `gmailr` packages as backend.
### 6.1. Using `emayili` (*RegLogEmayiliConnector*)
This backend is recommended to use. It supports many SMTP servers, mostly
with username and password based identification.
```{r, eval = F}
server <- function(input, output, session) {
mailConnector <- RegLogEmayiliConnector$new(
from = "[email protected]",
# to learn how to setup emayili smtp server read ?emayili::server
smtp = emayili::server(...)
)
}
```
### 6.2. Using `gmailr` (*RegLogGmailrConnector*)
This backend is only viable if you have an app registered in Google Cloud Console.
It authorizes and sends email via gmail REST API, needing Oauth authorization
with high scopes.
```{r, eval = F}
server <- function(input, output, session) {
mailConnector <- RegLogGmailrConnector$new(
from = "[email protected]"
)
}
```
## 7. Setup *RegLogServer*
All of **RegLog system** is generated and maintained by the object of class
`RegLogSystem` in unison with *dbConnector* and *mailConnector* of your
choosing.
Its setup is pretty straightforward:
```{r, eval = F}
server <- function(input, output, session) {
RegLog <- RegLogServer$new(
# both of these elements need to be defined firstly or in this call
dbConnector = dbConnector,
mailConnector = mailConnector
)
}
```
Besides these two mandatory arguments, there are also some additional arguments
to be used for customization.
- `app_name`: your application name. It will be used within emails send to users.
If not specified, the name of the folder containing you application files will be
used.
- `app_address`: URL address to your application. It will be used within emails
send to users. If not specified, the email address contained within *session$clientData*
will be used.
- `lang`: language to be used in UI elements and send emails. Defaults to 'en'
for *English*. Currently also 'pl' for *Polish* is supported. Besides that, you
can also specify 'i18' to generate only text indicators - external methods can
be used to parse the message in given language can be then used (eg. `shiny.i18n`
package).
- `custom_txts`: named list with character strings of custom texts to be used
by the system. For more information call `?RegLog_txt`.
- `use_modals`: either boolean indicating if all default modalDialogs should be shown
or named list of `FALSE` values inhibiting specific ones. For more information
read **RegLogServer object fields and methods**, section **Message**. All modals
are listed there alongside conditions of their appearance.
- `module_id`: character string with ID of the module. Recommended to leave
as default - unless if that name is taken by another module.
### 7.1. Retrieve information from **RegLogServer**
After setting up and assigning the object your application logic can observe
status of user in current session by public fields containing *reactiveVal*
objects.
```{r, eval = F}
## if you assigned the RegLogServer to 'RegLog' object, as in examples above:
# boolean showing if the user is logged in:
RegLog$is_logged()
# character vector containing user ID: either specific to the user if logged
# in, or unique anonymous ID generated with 'uuid::UUIDgenerate()'
RegLog$user_id()
# character vector containing user email if logged in
RegLog$user_mail()
# integer of the account ID inside the database - for identifying the logged user
# across relative tables
RegLog$account_id()
```
>There are much more to be learned about RegLogServer object - for more information
read **RegLogServer object fields and methods**. Above information is enough
for basic setup.
## 8. Insert UI elements
Every part of the UI is generated by *RegLogServer*, and could be accessed by
provided functions containing `tagList`.
### 8.1. Login UI
Providing GUI to allow logging in if user is already registered to your
application.
```{r, eval = F}
RegLog_login_UI()
```
<img src="man/figures/login_UI.png" style="filter: drop-shadow(5px 5px 5px black); margin-bottom: 5px;">
### 8.2. Register UI
Providing GUI for registering new account.
- Both user ID and e-mail needs to be unique within the databasa
- user ID need to consist of 8-30 alphanumeric characters
- user password need to consist of 8-30 characters. It needs to contain characters
of at least 3 of 4 following groups:
- small letter
- big letter
- number
- special character
- user email is also validated
After account registration, user will receive confirmation email on their password.
```{r, eval = F}
RegLog_register_UI()
```
<img src="man/figures/register_UI.png" style="filter: drop-shadow(5px 5px 5px black); margin-bottom: 5px;">
### 8.3. Credentials edit UI
Providing GUI for changing credentials.
- user need to be logged in to change their data
- to confirm their identity, they need to provide their current password
- user can change either during one interaction:
- password
- user ID and/or e-mail
- same checks as in register procedure are conducted
- after user ID and/or e-mail change user will receive confirmation e-mail
```{r, eval = F}
RegLog_credsEdit_UI()
```
<img src="man/figures/credsEdit_UI.png" style="filter: drop-shadow(5px 5px 5px black); margin-bottom: 5px;">
### 8.4. Reset password UI
Providing GUI for password reset.
- in the case of user forgetting their password, they can reset it by providing
their user ID
- mail message containing reset code will be sent to the user
- inputing the reset code from e-mail will allow password change
- same password check for new password is conducted as in register procedure
```{r, eval = F}
RegLog_resetPass_UI()
```
<img src="man/figures/resetPass_UI.png" style="filter: drop-shadow(5px 5px 5px black); margin-bottom: 5px;">