Initial Lucee Rabbit CRUD UI
This commit is contained in:
commit
f510028ae1
126
Application.cfc
Normal file
126
Application.cfc
Normal file
@ -0,0 +1,126 @@
|
||||
<cfcomponent displayname="Application" output="true">
|
||||
<cfset this.Name = "rabbit-lsd-ui" />
|
||||
<cfset this.sessionmanagement = "Yes" />
|
||||
<cfset this.datasource = "testds" />
|
||||
<cfset this.javaSettings = { loadPaths = [ expandPath("./lib") ] } />
|
||||
<cfset getDS(this.datasource) />
|
||||
|
||||
<cffunction name="getEnv" access="private" returntype="string">
|
||||
<cfargument name="key" type="string" required="true" />
|
||||
<cfargument name="def" type="string" required="true" />
|
||||
<cfset var system = createObject("java", "java.lang.System") />
|
||||
<cfset var val = system.getEnv(arguments.key) />
|
||||
<cfif isDefined("val") AND len(val)><cfreturn val /></cfif>
|
||||
<cfreturn arguments.def />
|
||||
</cffunction>
|
||||
|
||||
<cffunction name="getDS" access="private" returntype="void">
|
||||
<cfargument name="dsname" type="string" required="true" />
|
||||
<cfset var system = createObject("java", "java.lang.System") />
|
||||
<cfset var ds = {} />
|
||||
<cfloop list="class,connectionString,database,driver,host,port,type,url,username,password,bundleName,bundleVersion,connectionLimit,liveTimeout,validate" item="field">
|
||||
<cfset var envVal = system.getEnv("#arguments.dsname#_#field#") />
|
||||
<cfif isDefined("envVal") AND len(envVal)><cfset ds[field] = envVal /></cfif>
|
||||
</cfloop>
|
||||
<cfset this.datasources[arguments.dsname] = ds />
|
||||
</cffunction>
|
||||
|
||||
<cffunction name="publishToRabbit" access="private" returntype="void">
|
||||
<cfargument name="payload" type="struct" required="true" />
|
||||
<cfset var host = getEnv("RABBIT_HOST", "") />
|
||||
<cfset var user = getEnv("RABBIT_USER", "") />
|
||||
<cfset var pass = getEnv("RABBIT_PASSWORD", "") />
|
||||
<cfset var vhost = getEnv("RABBIT_VHOST", "/") />
|
||||
<cfset var port = getEnv("RABBIT_PORT", "5672") />
|
||||
<cfset var queueList = listToArray(getEnv("RABBIT_QUEUES", "crud_queue")) />
|
||||
<cfset var queueName = trim(queueList[1]) />
|
||||
<cfset var durable = lcase(getEnv("RABBIT_DURABLE", "true")) EQ "true" />
|
||||
<cfset var factory = createObject("java", "com.rabbitmq.client.ConnectionFactory") />
|
||||
<cfset var connection = "" />
|
||||
<cfset var channel = "" />
|
||||
|
||||
<cfif host EQ "" OR user EQ "" OR pass EQ "">
|
||||
<cfthrow message="Rabbit credentials are missing" />
|
||||
</cfif>
|
||||
|
||||
<cfset factory.setHost(host) />
|
||||
<cfset factory.setPort(JavaCast("int", port)) />
|
||||
<cfset factory.setUsername(user) />
|
||||
<cfset factory.setPassword(pass) />
|
||||
<cfset factory.setVirtualHost(vhost) />
|
||||
|
||||
<cftry>
|
||||
<cfset connection = factory.newConnection() />
|
||||
<cfset channel = connection.createChannel() />
|
||||
<cfset channel.queueDeclare(queueName, durable, false, false, JavaCast("null", "")) />
|
||||
<cfset var payloadJson = serializeJSON(arguments.payload) />
|
||||
<cfset var body = createObject("java", "java.lang.String").init(payloadJson).getBytes("UTF-8") />
|
||||
<cfset channel.basicPublish("", queueName, JavaCast("null", ""), body) />
|
||||
<cfif isDefined("channel")><cfset channel.close() /></cfif>
|
||||
<cfif isDefined("connection")><cfset connection.close() /></cfif>
|
||||
<cfcatch>
|
||||
<cfif isDefined("channel")><cfset channel.close() /></cfif>
|
||||
<cfif isDefined("connection")><cfset connection.close() /></cfif>
|
||||
<cfthrow message="#cfcatch.message#" />
|
||||
</cfcatch>
|
||||
</cftry>
|
||||
</cffunction>
|
||||
|
||||
<cffunction name="OnRequest" access="public" returntype="void" output="true">
|
||||
<cfargument name="template" type="string" required="true" />
|
||||
<cfset request.DS = this.datasource />
|
||||
<cfset request.tableName = getEnv("PG_TABLE", "rabbit_messages") />
|
||||
|
||||
<cftry>
|
||||
<cfquery datasource="#request.DS#">
|
||||
CREATE TABLE IF NOT EXISTS #request.tableName# (id SERIAL PRIMARY KEY, test_data TEXT, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP)
|
||||
</cfquery>
|
||||
<cfcatch><cfset request.db_error = cfcatch.message /></cfcatch>
|
||||
</cftry>
|
||||
|
||||
<cfif CGI.REQUEST_METHOD EQ "POST" AND structKeyExists(form, "crud_action")>
|
||||
<cfset var action = lcase(form.crud_action) />
|
||||
<cfset var requestId = createUUID() />
|
||||
<cfset var payload = { "request_id" = requestId } />
|
||||
|
||||
<cftry>
|
||||
<cfswitch expression="#action#">
|
||||
<cfcase value="insert">
|
||||
<cfset payload.action = "create" />
|
||||
<cfset payload.text = trim(form.txt_content) & " [req:" & requestId & "]" />
|
||||
<cfset session.last_action = { action = "create", request_id = requestId, queued_at = now(), text = trim(form.txt_content) } />
|
||||
</cfcase>
|
||||
<cfcase value="update">
|
||||
<cfif NOT structKeyExists(form, "id")>
|
||||
<cfthrow message="Missing id for update" />
|
||||
</cfif>
|
||||
<cfset payload.action = "update" />
|
||||
<cfset payload.id = val(form.id) />
|
||||
<cfset payload.text = trim(form.txt_content) & " [req:" & requestId & "]" />
|
||||
<cfset session.last_action = { action = "update", request_id = requestId, queued_at = now(), id = val(form.id), text = trim(form.txt_content) } />
|
||||
</cfcase>
|
||||
<cfcase value="delete">
|
||||
<cfif NOT structKeyExists(form, "id")>
|
||||
<cfthrow message="Missing id for delete" />
|
||||
</cfif>
|
||||
<cfset payload.action = "delete" />
|
||||
<cfset payload.id = val(form.id) />
|
||||
<cfset session.last_action = { action = "delete", request_id = requestId, queued_at = now(), id = val(form.id) } />
|
||||
</cfcase>
|
||||
<cfdefaultcase>
|
||||
<cfthrow message="Unknown action" />
|
||||
</cfdefaultcase>
|
||||
</cfswitch>
|
||||
|
||||
<cfset publishToRabbit(payload) />
|
||||
<cflocation url="#CGI.SCRIPT_NAME#" addtoken="false">
|
||||
<cfcatch>
|
||||
<cfset request.queue_error = cfcatch.message />
|
||||
<cfset structDelete(session, "last_action") />
|
||||
</cfcatch>
|
||||
</cftry>
|
||||
</cfif>
|
||||
|
||||
<cfinclude template="#arguments.template#" />
|
||||
</cffunction>
|
||||
</cfcomponent>
|
||||
38
README.md
Normal file
38
README.md
Normal file
@ -0,0 +1,38 @@
|
||||
# Lucee UI for Rabbit CRUD
|
||||
|
||||
UI publishes CRUD actions to RabbitMQ and reads results from Postgres.
|
||||
|
||||
## Behavior
|
||||
|
||||
- Insert/Update/Delete actions are queued via RabbitMQ.
|
||||
- A status banner shows when the action was queued and when it appears in Postgres.
|
||||
- The UI reads the latest rows from the table defined by `PG_TABLE`.
|
||||
|
||||
## Environment variables
|
||||
|
||||
Lucee datasource (required):
|
||||
- `testds_connectionString`
|
||||
- `testds_username`
|
||||
- `testds_password`
|
||||
- `testds_class` (org.postgresql.Driver)
|
||||
- `testds_bundleName` (org.postgresql.jdbc)
|
||||
- `testds_bundleVersion` (42.6.0)
|
||||
- `testds_connectionLimit` (5)
|
||||
- `testds_liveTimeout` (15)
|
||||
- `testds_validate` (false)
|
||||
|
||||
RabbitMQ (required):
|
||||
- `RABBIT_HOST`
|
||||
- `RABBIT_PORT` (default 5672)
|
||||
- `RABBIT_USER`
|
||||
- `RABBIT_PASSWORD`
|
||||
- `RABBIT_VHOST` (default `/`)
|
||||
- `RABBIT_QUEUES` (default `crud_queue`)
|
||||
- `RABBIT_DURABLE` (default `true`)
|
||||
|
||||
Optional:
|
||||
- `PG_TABLE` (default `rabbit_messages`)
|
||||
|
||||
## Notes
|
||||
|
||||
This app uses RabbitMQ Java client JARs located in `lib/`.
|
||||
BIN
lib/amqp-client-5.21.0.jar
Normal file
BIN
lib/amqp-client-5.21.0.jar
Normal file
Binary file not shown.
BIN
lib/slf4j-api-1.7.36.jar
Normal file
BIN
lib/slf4j-api-1.7.36.jar
Normal file
Binary file not shown.
117
query.cfm
Normal file
117
query.cfm
Normal file
@ -0,0 +1,117 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>Nubes | Rabbit CRUD UI</title>
|
||||
<link rel="icon" href="https://nubes.ru/themes/custom/nubes/images/nubes-ico.svg" type="image/svg+xml">
|
||||
<style>
|
||||
:root { --nubes-blue: #005BFF; --nubes-dark: #1A1A1A; --nubes-grey: #F8F9FA; --nubes-border: #E5E7EB; --nubes-green: #0f9d58; --nubes-red: #d93025; }
|
||||
body { font-family: 'Segoe UI', Tahoma, sans-serif; margin: 0; padding: 0; background: var(--nubes-grey); color: var(--nubes-dark); }
|
||||
.header-bg { position: sticky; top: 0; z-index: 1000; background: #fff; border-bottom: 1px solid var(--nubes-border); padding: 15px 0; box-shadow: 0 2px 10px rgba(0,0,0,0.05); }
|
||||
.container { max-width: 1000px; margin: auto; padding: 0 20px; }
|
||||
.header-content { display: flex; align-items: center; justify-content: space-between; }
|
||||
.logo { height: 40px; }
|
||||
.main-content { padding: 40px 0; }
|
||||
.card { background: #fff; padding: 32px; border-radius: 16px; box-shadow: 0 4px 20px rgba(0,0,0,0.04); }
|
||||
.btn { display: inline-flex; align-items: center; justify-content: center; cursor: pointer; padding: 12px 24px; border: none; border-radius: 8px; font-weight: 600; font-size: 14px; }
|
||||
.btn-primary { background: var(--nubes-blue); color: #fff; }
|
||||
.btn-action { padding: 12px; background: #fff; border: 1px solid var(--nubes-border); border-radius: 8px; font-size: 24px; line-height: 1; cursor: pointer; min-width: 50px; }
|
||||
.btn-action:hover { background: var(--nubes-grey); border-color: var(--nubes-blue); }
|
||||
.input-group { display: flex; gap: 12px; margin-bottom: 16px; }
|
||||
input[type="text"] { flex-grow: 1; padding: 12px 16px; border: 1px solid var(--nubes-border); border-radius: 8px; font-size: 14px; }
|
||||
table { width: 100%; border-collapse: collapse; }
|
||||
th { text-align: left; padding: 16px; font-size: 12px; text-transform: uppercase; color: #6B7280; border-bottom: 1px solid var(--nubes-border); }
|
||||
td { padding: 16px; border-bottom: 1px solid var(--nubes-border); }
|
||||
tbody tr:nth-child(even) { background-color: #FAFBFC; }
|
||||
tbody tr:hover { background-color: #F3F4F6; }
|
||||
.id-cell { font-family: monospace; color: #9CA3AF; width: 60px; }
|
||||
.actions-cell { display: flex; gap: 12px; width: 130px; }
|
||||
.alert { padding: 12px 16px; border-radius: 10px; margin-bottom: 20px; font-size: 14px; }
|
||||
.alert-info { background: #eef4ff; color: #234; border: 1px solid #d6e3ff; }
|
||||
.alert-success { background: #eef8f1; color: #1e4620; border: 1px solid #cbe6d3; }
|
||||
.alert-error { background: #fdeeee; color: var(--nubes-red); border: 1px solid #f4c7c3; }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<cfset status = {}>
|
||||
<cfif structKeyExists(request, "queue_error")>
|
||||
<cfset status.type = "error">
|
||||
<cfset status.message = "Ошибка постановки в очередь: " & request.queue_error>
|
||||
<cfelseif structKeyExists(session, "last_action")>
|
||||
<cfset la = session.last_action>
|
||||
<cfset status.type = "info">
|
||||
<cfset status.message = "Поставлено в очередь: " & uCase(la.action) & " в " & dateTimeFormat(la.queued_at, "yyyy-mm-dd HH:nn:ss")>
|
||||
<cfif la.action EQ "delete">
|
||||
<cfquery name="qCheck" datasource="#request.DS#">
|
||||
SELECT id FROM #request.tableName# WHERE id = <cfqueryparam value="#la.id#" cfsqltype="cf_sql_integer">
|
||||
</cfquery>
|
||||
<cfif qCheck.recordCount EQ 0>
|
||||
<cfset status.type = "success">
|
||||
<cfset status.message = "Исполнено в " & dateTimeFormat(now(), "yyyy-mm-dd HH:nn:ss")>
|
||||
<cfset structDelete(session, "last_action")>
|
||||
</cfif>
|
||||
<cfelse>
|
||||
<cfset reqToken = "[req:" & la.request_id & "]">
|
||||
<cfquery name="qCheck" datasource="#request.DS#">
|
||||
SELECT id, created_at FROM #request.tableName# WHERE test_data LIKE <cfqueryparam value="%#reqToken#%" cfsqltype="cf_sql_varchar"> ORDER BY id DESC LIMIT 1
|
||||
</cfquery>
|
||||
<cfif qCheck.recordCount GT 0>
|
||||
<cfset status.type = "success">
|
||||
<cfset status.message = "Исполнено в " & dateTimeFormat(qCheck.created_at, "yyyy-mm-dd HH:nn:ss")>
|
||||
<cfset structDelete(session, "last_action")>
|
||||
</cfif>
|
||||
</cfif>
|
||||
</cfif>
|
||||
|
||||
<div class="header-bg">
|
||||
<div class="container header-content">
|
||||
<img src="https://nubes.ru/themes/custom/nubes_2025/logo.svg" alt="Nubes" class="logo">
|
||||
<div style="font-size: 14px; color: var(--nubes-blue); font-weight: 600;">Lucee + Rabbit + Postgres Demo</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="container main-content">
|
||||
<div class="card">
|
||||
<cfif structKeyExists(request, "db_error")>
|
||||
<div class="alert alert-error">Ошибка базы данных: <cfoutput>#request.db_error#</cfoutput></div>
|
||||
</cfif>
|
||||
<cfif structKeyExists(status, "message")>
|
||||
<div class="alert <cfoutput>#status.type EQ 'success' ? 'alert-success' : (status.type EQ 'error' ? 'alert-error' : 'alert-info')#</cfoutput>">
|
||||
<cfoutput>#status.message#</cfoutput>
|
||||
</div>
|
||||
</cfif>
|
||||
|
||||
<form method="post" class="input-group">
|
||||
<input type="hidden" name="crud_action" value="insert">
|
||||
<input type="text" name="txt_content" placeholder="Новое сообщение..." required>
|
||||
<button type="submit" class="btn btn-primary">Поставить в очередь</button>
|
||||
</form>
|
||||
|
||||
<cfquery name="qGet" datasource="#request.DS#">SELECT * FROM #request.tableName# ORDER BY id DESC LIMIT 20</cfquery>
|
||||
<table>
|
||||
<thead><tr><th>ID</th><th>Содержимое</th><th>Действия</th></tr></thead>
|
||||
<tbody>
|
||||
<cfoutput query="qGet">
|
||||
<cfset cleanText = reReplace(test_data, "(?i)\\s*\\[req:[0-9a-f-]+\\]$", "", "all")>
|
||||
<tr>
|
||||
<td class="id-cell">#id#</td>
|
||||
<td>
|
||||
<form method="post" id="upd_#id#" style="margin:0">
|
||||
<input type="hidden" name="crud_action" value="update"><input type="hidden" name="id" value="#id#">
|
||||
<input type="text" name="txt_content" value="#HTMLEditFormat(cleanText)#" style="width:100%; border:none; background:transparent;">
|
||||
</form>
|
||||
</td>
|
||||
<td class="actions-cell">
|
||||
<button type="submit" form="upd_#id#" class="btn-action">💾</button>
|
||||
<form method="post" style="margin:0" onsubmit="return confirm('Удалить?')">
|
||||
<input type="hidden" name="crud_action" value="delete"><input type="hidden" name="id" value="#id#">
|
||||
<button type="submit" class="btn-action">🗑</button>
|
||||
</form>
|
||||
</td>
|
||||
</tr>
|
||||
</cfoutput>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
Loading…
Reference in New Issue
Block a user