-
Notifications
You must be signed in to change notification settings - Fork 15
/
initfuncs.R
75 lines (61 loc) · 2.18 KB
/
initfuncs.R
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
listen_for_authcode <- function(remote_url, local_url)
{
local_url <- httr::parse_url(local_url)
localhost <- if(local_url$hostname == "localhost") "127.0.0.1" else local_url$hostname
localport <- local_url$port
# based on httr::oauth_listener
info <- NULL
listen <- function(env)
{
query <- env$QUERY_STRING
info <<- if(is.character(query) && nchar(query) > 0)
httr::parse_url(query)$query
else list()
if(is_empty(info$code))
list(status=404L, headers=list(`Content-Type`="text/plain"), body="Not found")
else list(status=200L, headers=list(`Content-Type`="text/plain"),
body="Authenticated with Azure Active Directory. Please close this page and return to R.")
}
server <- httpuv::startServer(as.character(localhost), as.integer(localport), list(call=listen))
on.exit(httpuv::stopServer(server))
message("Waiting for authentication in browser...\nPress Esc/Ctrl + C to abort")
httr::BROWSE(remote_url)
while(is.null(info))
{
httpuv::service()
Sys.sleep(0.001)
}
httpuv::service()
if(is_empty(info$code))
{
msg <- gsub("\\+", " ", utils::URLdecode(info$error_description))
stop("Authentication failed. Message:\n", msg, call.=FALSE)
}
message("Authentication complete.")
info$code
}
poll_for_token <- function(url, body, interval, period)
{
interval <- as.numeric(interval)
ntries <- as.numeric(period) %/% interval
body$grant_type <- "urn:ietf:params:oauth:grant-type:device_code"
message("Waiting for device code in browser...\nPress Esc/Ctrl + C to abort")
for(i in seq_len(ntries))
{
Sys.sleep(interval)
res <- httr::POST(url, body=body, encode="form")
status <- httr::status_code(res)
cont <- httr::content(res)
if(status == 400 && cont$error == "authorization_pending")
{
# do nothing
}
else if(status >= 300)
process_aad_response(res) # fail here on error
else break
}
if(status >= 300)
stop("Authentication failed.", call.=FALSE)
message("Authentication complete.")
res
}