Bring data into Shiny on the same port π
Seamlessly integrate POST requests and query parameters into your existing Shiny apps as reactive values. Zero configuration - works with your current UI on the same port.
β¨ Key Features
- π Same-port integration - No need for separate servers or ports
- π‘ RESTful API endpoints - Accept POST requests alongside your Shiny UI
- π Reactive data streams - POST data automatically becomes reactive values
- π‘οΈ Built-in authentication - Token-based security for your endpoints
- π± Multiple data formats - JSON, form data, query parameters
- π Cross-session sharing - Data shared across all connected clients
- π¦ Production ready - Comprehensive testing and CRAN-quality code
π Quick Start
Installation
# Install from CRAN (when available)
install.packages("shinypayload")
# Install development version from GitHub
remotes::install_github("PawanRamaMali/shinypayload")
# For local development
devtools::load_all()
Basic Example
library(shiny)
library(shinypayload)
# Your regular UI - no changes needed!
base_ui <- fluidPage(
titlePanel("π shinypayload Demo"),
fluidRow(
column(6,
h4("π Live Data"),
verbatimTextOutput("live_data")
),
column(6,
h4("π URL Parameters"),
verbatimTextOutput("url_params")
)
),
hr(),
h4("π‘ POST Endpoint"),
verbatimTextOutput("endpoint_info")
)
# Wrap your UI to handle POST requests
ui <- payload_ui(
base_ui,
path = "/api/data",
token = Sys.getenv("API_TOKEN", "demo-token")
)
server <- function(input, output, session) {
# Get URL parameters
output$url_params <- renderPrint({
params <- params_get(session)
if (length(params) > 0) params else "No URL parameters"
})
# Show endpoint URL
output$endpoint_info <- renderText({
url <- payload_endpoint_url(session, "/api/data")
paste("Send POST requests to:", url, "?token=demo-token")
})
# React to incoming POST data
live_data <- payload_last("/api/data", session, intervalMillis = 200)
output$live_data <- renderPrint({
data <- live_data()
if (is.null(data)) {
"Waiting for data... π‘"
} else {
list(
timestamp = data$meta$timestamp,
payload = data$payload,
source = data$meta$remote_addr
)
}
})
}
# IMPORTANT: Use uiPattern = ".*" for POST routing
shinyApp(ui, server, uiPattern = ".*")
Send Data to Your App
# Send JSON data
curl -X POST "http://localhost:3838/api/data?token=demo-token" \\
-H "Content-Type: application/json" \\
-d '{"sensor": "temperature", "value": 23.5, "unit": "celsius"}'
# Send form data
curl -X POST "http://localhost:3838/api/data?token=demo-token" \\
-d "name=sensor01&status=active&reading=42"
# Response: {"ok": true}
π Documentation
Core Functions
Function | Purpose | Example |
---|---|---|
payload_ui() |
Wrap UI to handle POST requests | payload_ui(my_ui, "/api", "token") |
payload_last() |
Get reactive with latest POST data | data <- payload_last("/api", session) |
params_get() |
Extract URL query parameters | params <- params_get(session) |
payload_endpoint_url() |
Get full endpoint URL | url <- payload_endpoint_url(session, "/api") |
Authentication Methods
# Query parameter (recommended)
POST /api/data?token=your-secret-token
# HTTP Headers
POST /api/data
X-Ingress-Token: your-secret-token
# OR
Authorization: your-secret-token
Supported Content Types
-
application/json
- Parsed withjsonlite::fromJSON()
-
application/x-www-form-urlencoded
- Parsed withshiny::parseQueryString()
- Fallback: Attempts JSON parsing, returns raw text if failed
π‘ Use Cases
π Complete Examples
Explore our comprehensive examples in inst/examples/
:
-
basic_example.R
- Core functionality demo -
real_time_monitor.R
- Live data monitoring -
form_handler.R
- Form submission processing
-
existing_app_integration.R
- Add to existing apps
Each example includes ready-to-run code and curl commands for testing.
π‘οΈ Security Best Practices
-
Always use tokens in production
ui <- payload_ui(base_ui, "/api", Sys.getenv("API_SECRET"))
-
Validate and sanitize input data
observeEvent(payload_last("/api", session), { data <- payload_last("/api", session)() if (!is.null(data)) { # Validate required fields if (is.null(data$payload$user_id)) return() # Sanitize inputs before use clean_data <- DBI::dbQuoteString(pool, data$payload$message) } })
Use HTTPS in production
Implement rate limiting if needed
Monitor and log API usage
π§ Advanced Configuration
Multiple Endpoints
# Handle different data types on different paths
ui <- payload_ui(base_ui, "/sensors", "sensor-token")
server <- function(input, output, session) {
# Different reactives for different endpoints
sensor_data <- payload_last("/sensors", session)
user_data <- payload_last("/users", session)
# Process accordingly...
}
Custom Polling Intervals
# High-frequency updates (100ms)
fast_data <- payload_last("/live", session, intervalMillis = 100)
# Low-frequency updates (5 seconds)
slow_data <- payload_last("/batch", session, intervalMillis = 5000)
π€ Contributing
We welcome contributions! Please see our Contributing Guidelines for details.
π Package Status
- β 132 tests with comprehensive coverage
- β Cross-platform compatibility (Windows, macOS, Linux)
- β Multiple R versions supported (R β₯ 4.1)
- β CRAN ready - passes all checks
- β Production tested - used in real applications
π License
This project is licensed under the MIT License - see the LICENSE file for details.
π Acknowledgments
- Built on the amazing Shiny framework
- Inspired by the need for seamless data integration in web applications
- Thanks to the R community for feedback and contributions
π Support
- π Documentation: Package website
- π Bug reports: GitHub Issues
- π¬ Questions: GitHub Issues
- π§ Email: prm@outlook.in
Made with β€οΈ for the R and Shiny community