Reputation: 2538
I have a shiny application that queries data from SQL into data frames, and then those data frames are referenced from my shinyServer() block. I've been running it only in RStudio thus far, and so whenever I needed new data I'd just restart the application and before the server loads it would grab all new data.
I'd like to transition the app to shiny server, but I'm not sure how I can induce it to get new data periodically. For the sake of the interface I'd like it to be automatic rather than have a user click a button to initiate the loading. Is there an idiomatic solution for this?
EDIT:
I think I found a solution that works for me.
shinyServer(function(input,output,session){
sourceData <- reactive({
invalidateLater(1000000,session)
functionThatGetsData()
})
})
Upvotes: 27
Views: 29567
Reputation: 121
I don't think either of these answers fully answers OP's question. Specifically
whenever I needed new data I'd just restart the application and before the server loads it would grab all new data.
which suggests that the global object needs to be refreshed. Using reactives inside the server function is not a great answer to that problem bc you have a race condition if multiple users are interacting with the app simultaneously ... all of them will try to update the object concurrently, which may or may not resolve itself well depending on what the update is (this is something I'm actively wrestling with).
One possible solution I've explored is using the flock package to have each user attempt to acquire a lock on a global temp file, and the user successfully acquiring it would be tasked with doing the global update. This directly addresses the race condition. I haven't tested it fully enough to post it as a solution here, maybe someone else can run with that.
Upvotes: 2
Reputation: 1232
The smartest would probable be to use reactivePoll
if you can make a fast query to detect if there is new data. This worked very well for me just today actually.
reactivePoll shiny
Reactive polling
Description
Used to create a reactive data source, which works by periodically polling a non-reactive data ource.
Usage
reactivePoll(intervalMillis, session, checkFunc, valueFunc)
Arguments
intervalMillis
Approximate number of milliseconds to wait between calls to
checkFunc
. his an be either a numeric value, or a function that returns a numeric value.session
The user session to associate this file reader with, or
NULL
if none. If non-null, he reader will automatically stop when the session ends.checkFunc
A relatively cheap function whose values over time will be tested for equality; nequality indicates that the underlying value has changed and needs to be invalidated and re-ead usingvalueFunc
. See Details.valueFunc
A function that calculates the underlying value. See Details.
Upvotes: 30
Reputation: 44614
You're looking for invalidateLater
. Put this, with the appropriate interval, in the reactive expression that retrieves data from the the database.
Upvotes: 10