panda-core 0.9.1 → 0.9.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/app/controllers/panda/core/admin/settings_controller.rb +13 -0
- data/app/helpers/panda/core/asset_helper.rb +27 -9
- data/app/views/panda/core/admin/settings/show.html.erb +10 -0
- data/app/views/panda/core/shared/_header.html.erb +5 -2
- data/config/routes.rb +3 -0
- data/lib/panda/core/asset_loader.rb +3 -2
- data/lib/panda/core/engine/middleware_config.rb +4 -20
- data/lib/panda/core/engine.rb +21 -0
- data/lib/panda/core/module_registry.rb +54 -0
- data/lib/panda/core/testing/support/system/ci_capybara_config.rb +57 -0
- data/lib/panda/core/version.rb +1 -1
- metadata +4 -1
checksums.yaml
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
---
|
|
2
2
|
SHA256:
|
|
3
|
-
metadata.gz:
|
|
4
|
-
data.tar.gz:
|
|
3
|
+
metadata.gz: '08a72a419dd702e8cc8cde2bfea476a283a17280499bb16de53b554ebaf7406c'
|
|
4
|
+
data.tar.gz: 4cf2468a51ed77691ab280b2cfb4b0345a2449e4bb61a36560b286cecb311cac
|
|
5
5
|
SHA512:
|
|
6
|
-
metadata.gz:
|
|
7
|
-
data.tar.gz:
|
|
6
|
+
metadata.gz: d8ebe8e076e53a063dc0f7ab565c6b12c299a6765cdd7e9c446716a5de191070ea2da46a75167f384ab4ee5c2fa91753543625e2cbc64f58a9f962725be39042
|
|
7
|
+
data.tar.gz: c7d7d8932c8eea8b767343e27a4ce23ac0206c40df9acd6557b5734b2684f16080a2586fc99a63d62b0b548427dd35cf4603219e2d9938f3d704d873da818a39
|
|
@@ -9,6 +9,10 @@ module Panda
|
|
|
9
9
|
end
|
|
10
10
|
|
|
11
11
|
# Include only Core JavaScript
|
|
12
|
+
#
|
|
13
|
+
# This is the single entry point for all Panda JavaScript loading.
|
|
14
|
+
# It automatically includes JavaScript from Core and all registered modules
|
|
15
|
+
# via ModuleRegistry.
|
|
12
16
|
def panda_core_javascript
|
|
13
17
|
# Use asset_tags for development mode (importmap) compatibility
|
|
14
18
|
# In development, this will use importmap; in production/test, compiled bundles
|
|
@@ -22,20 +26,34 @@ module Panda
|
|
|
22
26
|
javascript_include_tag(js_url, type: "module")
|
|
23
27
|
end
|
|
24
28
|
else
|
|
25
|
-
# Development mode - Use
|
|
26
|
-
|
|
27
|
-
# Build the importmap JSON manually since paths are already absolute
|
|
28
|
-
imports = {}
|
|
29
|
-
Panda::Core.importmap.instance_variable_get(:@packages).each do |name, package|
|
|
30
|
-
imports[name] = package[:path]
|
|
31
|
-
end
|
|
29
|
+
# Development mode - Use ModuleRegistry to combine all Panda module importmaps
|
|
30
|
+
imports = Panda::Core::ModuleRegistry.combined_importmap
|
|
32
31
|
|
|
33
32
|
importmap_json = JSON.generate({"imports" => imports})
|
|
34
33
|
|
|
34
|
+
# Generate entry point script tags for Core and all registered modules
|
|
35
|
+
entry_points = []
|
|
36
|
+
|
|
37
|
+
# Core entry points (always included)
|
|
38
|
+
entry_points << '<script type="module">import "panda/core/application"</script>'
|
|
39
|
+
entry_points << '<script type="module">import "panda/core/controllers/index"</script>'
|
|
40
|
+
|
|
41
|
+
# Add entry points for each registered module
|
|
42
|
+
Panda::Core::ModuleRegistry.modules.each do |gem_name, info|
|
|
43
|
+
# Extract module namespace from gem name (e.g., "panda-cms" -> "cms")
|
|
44
|
+
module_slug = gem_name.sub(/^panda-/, "")
|
|
45
|
+
|
|
46
|
+
# Check if the module is actually loaded
|
|
47
|
+
module_name = info[:engine].sub(/::Engine$/, "")
|
|
48
|
+
next unless Object.const_defined?(module_name)
|
|
49
|
+
|
|
50
|
+
entry_points << %(<script type="module">import "panda/#{module_slug}/application"</script>)
|
|
51
|
+
entry_points << %(<script type="module">import "panda/#{module_slug}/controllers/index"</script>)
|
|
52
|
+
end
|
|
53
|
+
|
|
35
54
|
<<~HTML.html_safe
|
|
36
55
|
<script type="importmap">#{importmap_json}</script>
|
|
37
|
-
|
|
38
|
-
<script type="module">import "panda/core/controllers/index"</script>
|
|
56
|
+
#{entry_points.join("\n")}
|
|
39
57
|
HTML
|
|
40
58
|
end
|
|
41
59
|
end
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
<%= render Panda::Core::Admin::ContainerComponent.new do %>
|
|
2
|
+
<%= render Panda::Core::Admin::HeadingComponent.new(level: 1, text: "Settings") %>
|
|
3
|
+
|
|
4
|
+
<div class="mt-6 space-y-6">
|
|
5
|
+
<%= render Panda::Core::Admin::PanelComponent.new do %>
|
|
6
|
+
<%= render Panda::Core::Admin::HeadingComponent.new(level: 2, text: "Site Settings") %>
|
|
7
|
+
<p class="text-sm text-gray-600 mt-2">Configure your site settings here.</p>
|
|
8
|
+
<% end %>
|
|
9
|
+
</div>
|
|
10
|
+
<% end %>
|
|
@@ -6,11 +6,14 @@
|
|
|
6
6
|
<%= csp_meta_tag %>
|
|
7
7
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@7.1.0/css/all.min.css">
|
|
8
8
|
<%= panda_core_stylesheet %>
|
|
9
|
+
|
|
10
|
+
<!-- ES Module Shims polyfill - must load BEFORE importmap (blocking, not async) -->
|
|
11
|
+
<script src="https://ga.jspm.io/npm:[email protected]/dist/es-module-shims.js" crossorigin="anonymous"></script>
|
|
12
|
+
|
|
13
|
+
<!-- Panda JavaScript (Core + all registered modules via ModuleRegistry) -->
|
|
9
14
|
<%= panda_core_javascript %>
|
|
10
15
|
|
|
11
16
|
<% if defined?(Panda::CMS) && controller.class.name.start_with?("Panda::CMS") %>
|
|
12
|
-
<!-- CMS Assets -->
|
|
13
|
-
<%= panda_cms_javascript %>
|
|
14
17
|
<%= render "panda/cms/shared/favicons" %>
|
|
15
18
|
<% end %>
|
|
16
19
|
|
data/config/routes.rb
CHANGED
|
@@ -19,6 +19,9 @@ Panda::Core::Engine.routes.draw do
|
|
|
19
19
|
# Profile management
|
|
20
20
|
resource :my_profile, only: %i[show edit update], controller: "admin/my_profile", path: "my_profile"
|
|
21
21
|
|
|
22
|
+
# Settings
|
|
23
|
+
resource :settings, only: [:show], controller: "admin/settings"
|
|
24
|
+
|
|
22
25
|
# Test-only authentication endpoint (available in development and test environments)
|
|
23
26
|
# This bypasses OAuth for faster, more reliable test execution
|
|
24
27
|
# Development: Used by Capybara system tests which run Rails server in development mode
|
|
@@ -151,8 +151,9 @@ module Panda
|
|
|
151
151
|
version = asset_version
|
|
152
152
|
"/panda-core-assets/panda-core-#{version}.js"
|
|
153
153
|
else
|
|
154
|
-
# Return
|
|
155
|
-
|
|
154
|
+
# Return path for development/test mode
|
|
155
|
+
# JavaScript is served via importmap from app/javascript
|
|
156
|
+
"/panda/core/application.js"
|
|
156
157
|
end
|
|
157
158
|
end
|
|
158
159
|
|
|
@@ -4,29 +4,13 @@ module Panda
|
|
|
4
4
|
module Core
|
|
5
5
|
class Engine < ::Rails::Engine
|
|
6
6
|
# Middleware configuration for static assets
|
|
7
|
+
# Note: Actual middleware is registered in engine.rb initializer
|
|
8
|
+
# This module is kept for organizational purposes
|
|
7
9
|
module MiddlewareConfig
|
|
8
10
|
extend ActiveSupport::Concern
|
|
9
11
|
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
config.middleware.use Rack::Static,
|
|
13
|
-
urls: ["/panda-core-assets"],
|
|
14
|
-
root: Panda::Core::Engine.root.join("public"),
|
|
15
|
-
header_rules: [
|
|
16
|
-
# Disable caching in development for instant CSS updates
|
|
17
|
-
[:all, {"Cache-Control" => Rails.env.development? ? "no-cache, no-store, must-revalidate" : "public, max-age=31536000"}]
|
|
18
|
-
]
|
|
19
|
-
|
|
20
|
-
# Make JavaScript files available for importmap
|
|
21
|
-
# Serve from app/javascript with proper MIME types
|
|
22
|
-
config.middleware.use Rack::Static,
|
|
23
|
-
urls: ["/panda", "/panda/core"],
|
|
24
|
-
root: Panda::Core::Engine.root.join("app/javascript"),
|
|
25
|
-
header_rules: [
|
|
26
|
-
[:all, {"Cache-Control" => Rails.env.development? ? "no-cache, no-store, must-revalidate" : "public, max-age=31536000",
|
|
27
|
-
"Content-Type" => "text/javascript; charset=utf-8"}]
|
|
28
|
-
]
|
|
29
|
-
end
|
|
12
|
+
# Middleware configuration moved to engine.rb initializer
|
|
13
|
+
# See lib/panda/core/engine.rb - initializer "panda.core.static_assets"
|
|
30
14
|
end
|
|
31
15
|
end
|
|
32
16
|
end
|
data/lib/panda/core/engine.rb
CHANGED
|
@@ -57,6 +57,27 @@ module Panda
|
|
|
57
57
|
# Configuration is already initialized with defaults in Configuration class
|
|
58
58
|
end
|
|
59
59
|
|
|
60
|
+
# Static asset middleware for serving public files and JavaScript modules
|
|
61
|
+
initializer "panda.core.static_assets", before: :build_middleware_stack do |app|
|
|
62
|
+
# Make files in public available to the main app (e.g. /panda-core-assets/panda-logo.png)
|
|
63
|
+
app.config.middleware.use Rack::Static,
|
|
64
|
+
urls: ["/panda-core-assets"],
|
|
65
|
+
root: Panda::Core::Engine.root.join("public"),
|
|
66
|
+
header_rules: [
|
|
67
|
+
# Disable caching in development for instant CSS updates
|
|
68
|
+
[:all, {"Cache-Control" => Rails.env.development? ? "no-cache, no-store, must-revalidate" : "public, max-age=31536000"}]
|
|
69
|
+
]
|
|
70
|
+
|
|
71
|
+
# Make JavaScript files available for importmap
|
|
72
|
+
# Serve from app/javascript with proper MIME types
|
|
73
|
+
app.config.middleware.use Rack::Static,
|
|
74
|
+
urls: ["/panda/core"],
|
|
75
|
+
root: Panda::Core::Engine.root.join("app/javascript"),
|
|
76
|
+
header_rules: [
|
|
77
|
+
[:all, {"Cache-Control" => Rails.env.development? ? "no-cache, no-store, must-revalidate" : "public, max-age=31536000"}]
|
|
78
|
+
]
|
|
79
|
+
end
|
|
80
|
+
|
|
60
81
|
# Auto-compile CSS for test/development environments
|
|
61
82
|
initializer "panda_core.auto_compile_assets", after: :load_config_initializers do |app|
|
|
62
83
|
# Only auto-compile in test or when explicitly requested
|
|
@@ -144,8 +144,62 @@ module Panda
|
|
|
144
144
|
@modules.key?(gem_name)
|
|
145
145
|
end
|
|
146
146
|
|
|
147
|
+
# Returns a combined importmap for all registered modules
|
|
148
|
+
#
|
|
149
|
+
# This merges importmaps from Core and all registered modules (CMS, CMS Pro, etc.)
|
|
150
|
+
# into a single hash suitable for <script type="importmap"> generation.
|
|
151
|
+
#
|
|
152
|
+
# Order matters: Core imports are added first, then modules in registration order.
|
|
153
|
+
# If there are conflicts, later modules override earlier ones.
|
|
154
|
+
#
|
|
155
|
+
# @return [Hash] Combined imports hash {"module/name" => "/path/to/file.js"}
|
|
156
|
+
def combined_importmap
|
|
157
|
+
imports = {}
|
|
158
|
+
|
|
159
|
+
# Add Panda Core imports first (if Core has an importmap)
|
|
160
|
+
if defined?(Panda::Core.importmap)
|
|
161
|
+
Panda::Core.importmap.instance_variable_get(:@packages).each do |name, package|
|
|
162
|
+
imports[name] = package.path
|
|
163
|
+
end
|
|
164
|
+
end
|
|
165
|
+
|
|
166
|
+
# Add registered modules' importmaps in registration order
|
|
167
|
+
@modules.each do |gem_name, info|
|
|
168
|
+
next unless engine_available?(info[:engine])
|
|
169
|
+
|
|
170
|
+
# Get the module's importmap constant (e.g., Panda::CMS.importmap)
|
|
171
|
+
module_importmap = module_importmap_for(info[:engine])
|
|
172
|
+
next unless module_importmap
|
|
173
|
+
|
|
174
|
+
module_importmap.instance_variable_get(:@packages).each do |name, package|
|
|
175
|
+
imports[name] = package.path
|
|
176
|
+
end
|
|
177
|
+
end
|
|
178
|
+
|
|
179
|
+
imports
|
|
180
|
+
end
|
|
181
|
+
|
|
147
182
|
private
|
|
148
183
|
|
|
184
|
+
# Get the importmap for a module
|
|
185
|
+
#
|
|
186
|
+
# @param engine_name [String] Engine constant name (e.g., "Panda::CMS::Engine")
|
|
187
|
+
# @return [Importmap::Map, nil] Module's importmap or nil if not available
|
|
188
|
+
def module_importmap_for(engine_name)
|
|
189
|
+
# Extract module namespace from engine name (e.g., "Panda::CMS::Engine" -> "Panda::CMS")
|
|
190
|
+
module_name = engine_name.sub(/::Engine$/, "")
|
|
191
|
+
return nil unless Object.const_defined?(module_name)
|
|
192
|
+
|
|
193
|
+
mod = Object.const_get(module_name)
|
|
194
|
+
return nil unless mod.respond_to?(:importmap)
|
|
195
|
+
|
|
196
|
+
mod.importmap
|
|
197
|
+
rescue NoMethodError
|
|
198
|
+
nil
|
|
199
|
+
rescue NameError
|
|
200
|
+
nil
|
|
201
|
+
end
|
|
202
|
+
|
|
149
203
|
# Check if an engine constant is defined and available
|
|
150
204
|
#
|
|
151
205
|
# @param engine_name [String] Engine constant name
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
# CI-specific Capybara configuration
|
|
4
|
+
# This file configures Capybara for GitHub Actions and other CI environments
|
|
5
|
+
# It uses a more robust Puma setup with compatibility for Puma 6 and 7+
|
|
6
|
+
|
|
7
|
+
return unless defined?(Capybara)
|
|
8
|
+
|
|
9
|
+
ci_mode = ENV["GITHUB_ACTIONS"] == "true" || ENV["CI_SYSTEM_SPECS"] == "true"
|
|
10
|
+
return unless ci_mode
|
|
11
|
+
|
|
12
|
+
require "rack/handler/puma"
|
|
13
|
+
|
|
14
|
+
RSpec.configure do |config|
|
|
15
|
+
config.before(:suite) do
|
|
16
|
+
Capybara.server = :puma_ci
|
|
17
|
+
Capybara.default_max_wait_time = Integer(ENV.fetch("CAPYBARA_MAX_WAIT_TIME", 5))
|
|
18
|
+
Capybara.server_host = "127.0.0.1"
|
|
19
|
+
Capybara.always_include_port = true
|
|
20
|
+
|
|
21
|
+
puts "[CI Config] Capybara.server = #{Capybara.server.inspect}"
|
|
22
|
+
puts "[CI Config] Capybara.server_host = #{Capybara.server_host.inspect}"
|
|
23
|
+
puts "[CI Config] Capybara.max_wait = #{Capybara.default_max_wait_time}s"
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
Capybara.register_server :puma_ci do |app, port, host|
|
|
28
|
+
puts "[CI Config] Starting Puma (single mode) on #{host}:#{port}"
|
|
29
|
+
|
|
30
|
+
min_threads = Integer(ENV.fetch("PUMA_MIN_THREADS", "2"))
|
|
31
|
+
max_threads = Integer(ENV.fetch("PUMA_MAX_THREADS", "2"))
|
|
32
|
+
|
|
33
|
+
options = {
|
|
34
|
+
Host: host,
|
|
35
|
+
Port: port,
|
|
36
|
+
Threads: "#{min_threads}:#{max_threads}",
|
|
37
|
+
Workers: 0,
|
|
38
|
+
Silent: false,
|
|
39
|
+
Verbose: true,
|
|
40
|
+
PreloadApp: false
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
# --- Puma Compatibility Layer (supports Puma 6 AND Puma 7+) ---
|
|
44
|
+
puma_run = Rack::Handler::Puma.method(:run)
|
|
45
|
+
|
|
46
|
+
if puma_run.arity == 2
|
|
47
|
+
# Puma <= 6.x signature:
|
|
48
|
+
# run(app, options_hash)
|
|
49
|
+
puts "[CI Config] Using Puma <= 6 API (arity 2)"
|
|
50
|
+
Rack::Handler::Puma.run(app, options)
|
|
51
|
+
else
|
|
52
|
+
# Puma >= 7.x signature:
|
|
53
|
+
# run(app, **options)
|
|
54
|
+
puts "[CI Config] Using Puma >= 7 API (keyword args)"
|
|
55
|
+
Rack::Handler::Puma.run(app, **options)
|
|
56
|
+
end
|
|
57
|
+
end
|
data/lib/panda/core/version.rb
CHANGED
metadata
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: panda-core
|
|
3
3
|
version: !ruby/object:Gem::Version
|
|
4
|
-
version: 0.9.
|
|
4
|
+
version: 0.9.2
|
|
5
5
|
platform: ruby
|
|
6
6
|
authors:
|
|
7
7
|
- Otaina Limited
|
|
@@ -429,6 +429,7 @@ files:
|
|
|
429
429
|
- app/controllers/panda/core/admin/dashboard_controller.rb
|
|
430
430
|
- app/controllers/panda/core/admin/my_profile_controller.rb
|
|
431
431
|
- app/controllers/panda/core/admin/sessions_controller.rb
|
|
432
|
+
- app/controllers/panda/core/admin/settings_controller.rb
|
|
432
433
|
- app/controllers/panda/core/admin/test_sessions_controller.rb
|
|
433
434
|
- app/controllers/panda/core/application_controller.rb
|
|
434
435
|
- app/helpers/panda/core/asset_helper.rb
|
|
@@ -456,6 +457,7 @@ files:
|
|
|
456
457
|
- app/views/panda/core/admin/dashboard/show.html.erb
|
|
457
458
|
- app/views/panda/core/admin/my_profile/edit.html.erb
|
|
458
459
|
- app/views/panda/core/admin/sessions/new.html.erb
|
|
460
|
+
- app/views/panda/core/admin/settings/show.html.erb
|
|
459
461
|
- app/views/panda/core/admin/shared/_breadcrumbs.html.erb
|
|
460
462
|
- app/views/panda/core/admin/shared/_flash.html.erb
|
|
461
463
|
- app/views/panda/core/admin/shared/_sidebar.html.erb
|
|
@@ -506,6 +508,7 @@ files:
|
|
|
506
508
|
- lib/panda/core/testing/support/setup.rb
|
|
507
509
|
- lib/panda/core/testing/support/system/better_system_tests.rb
|
|
508
510
|
- lib/panda/core/testing/support/system/capybara_setup.rb
|
|
511
|
+
- lib/panda/core/testing/support/system/ci_capybara_config.rb
|
|
509
512
|
- lib/panda/core/testing/support/system/cuprite_helpers.rb
|
|
510
513
|
- lib/panda/core/testing/support/system/cuprite_setup.rb
|
|
511
514
|
- lib/panda/core/testing/support/system/database_connection_helpers.rb
|