smriti 0.5.0

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.
Files changed (124) hide show
  1. checksums.yaml +7 -0
  2. data/LICENSE +21 -0
  3. data/README.md +168 -0
  4. data/Rakefile +15 -0
  5. data/app/assets/images/smriti/android-chrome-192x192.png +0 -0
  6. data/app/assets/images/smriti/android-chrome-512x512.png +0 -0
  7. data/app/assets/images/smriti/apple-touch-icon.png +0 -0
  8. data/app/assets/images/smriti/favicon-16x16.png +0 -0
  9. data/app/assets/images/smriti/favicon-32x32.png +0 -0
  10. data/app/assets/images/smriti/favicon-48x48.png +0 -0
  11. data/app/assets/images/smriti/favicon.ico +0 -0
  12. data/app/assets/images/smriti/favicon.svg +18 -0
  13. data/app/assets/images/smriti/logo.svg +18 -0
  14. data/app/assets/images/smriti/mask-icon.svg +5 -0
  15. data/app/assets/stylesheets/smriti/application.css +1040 -0
  16. data/app/controllers/smriti/admin/application_controller.rb +135 -0
  17. data/app/controllers/smriti/admin/dashboard_controller.rb +32 -0
  18. data/app/controllers/smriti/admin/mat_view_definitions_controller.rb +372 -0
  19. data/app/controllers/smriti/admin/mat_view_runs_controller.rb +185 -0
  20. data/app/controllers/smriti/admin/preferences_controller.rb +91 -0
  21. data/app/helpers/smriti/admin/datatable_helper.rb +249 -0
  22. data/app/helpers/smriti/admin/localized_digit_helper.rb +70 -0
  23. data/app/helpers/smriti/admin/ui_helper.rb +539 -0
  24. data/app/javascript/smriti/application.js +8 -0
  25. data/app/javascript/smriti/controllers/application.js +10 -0
  26. data/app/javascript/smriti/controllers/body_setup_controller.js +120 -0
  27. data/app/javascript/smriti/controllers/datatable_controller.js +351 -0
  28. data/app/javascript/smriti/controllers/details_controller.js +200 -0
  29. data/app/javascript/smriti/controllers/drawer_controller.js +470 -0
  30. data/app/javascript/smriti/controllers/flash_controller.js +112 -0
  31. data/app/javascript/smriti/controllers/index.js +10 -0
  32. data/app/javascript/smriti/controllers/mv_confirm_controller.js +435 -0
  33. data/app/javascript/smriti/controllers/tabs_controller.js +184 -0
  34. data/app/javascript/smriti/controllers/tooltip_controller.js +525 -0
  35. data/app/javascript/smriti/controllers/turbo_frame_lifecycle_controller.js +342 -0
  36. data/app/jobs/smriti/application_job.rb +144 -0
  37. data/app/jobs/smriti/create_view_job.rb +87 -0
  38. data/app/jobs/smriti/delete_view_job.rb +89 -0
  39. data/app/jobs/smriti/refresh_view_job.rb +94 -0
  40. data/app/models/concerns/smriti_i18n.rb +139 -0
  41. data/app/models/concerns/smriti_paginate.rb +70 -0
  42. data/app/models/concerns/smriti_query_helper.rb +36 -0
  43. data/app/models/smriti/application_record.rb +39 -0
  44. data/app/models/smriti/mat_view_definition.rb +254 -0
  45. data/app/models/smriti/mat_view_run.rb +275 -0
  46. data/app/views/layouts/smriti/_footer.html.erb +47 -0
  47. data/app/views/layouts/smriti/_header.html.erb +25 -0
  48. data/app/views/layouts/smriti/admin.html.erb +47 -0
  49. data/app/views/layouts/smriti/turbo_frame.html.erb +3 -0
  50. data/app/views/smriti/admin/dashboard/index.html.erb +38 -0
  51. data/app/views/smriti/admin/mat_view_definitions/_definition_actions.html.erb +94 -0
  52. data/app/views/smriti/admin/mat_view_definitions/_dt-index-empty-row.html.erb +11 -0
  53. data/app/views/smriti/admin/mat_view_definitions/_dt-index-row.html.erb +27 -0
  54. data/app/views/smriti/admin/mat_view_definitions/empty.html.erb +1 -0
  55. data/app/views/smriti/admin/mat_view_definitions/form.html.erb +79 -0
  56. data/app/views/smriti/admin/mat_view_definitions/index.html.erb +10 -0
  57. data/app/views/smriti/admin/mat_view_definitions/show.html.erb +40 -0
  58. data/app/views/smriti/admin/mat_view_runs/_dt-index-empty-row.html.erb +11 -0
  59. data/app/views/smriti/admin/mat_view_runs/_dt-index-row.html.erb +41 -0
  60. data/app/views/smriti/admin/mat_view_runs/index.html.erb +1 -0
  61. data/app/views/smriti/admin/mat_view_runs/show.html.erb +64 -0
  62. data/app/views/smriti/admin/preferences/show.html.erb +49 -0
  63. data/app/views/smriti/admin/ui/_card.html.erb +15 -0
  64. data/app/views/smriti/admin/ui/_datatable.html.erb +34 -0
  65. data/app/views/smriti/admin/ui/_datatable_filters.html.erb +45 -0
  66. data/app/views/smriti/admin/ui/_datatable_tbody.html.erb +11 -0
  67. data/app/views/smriti/admin/ui/_datatable_tfoot.html.erb +70 -0
  68. data/app/views/smriti/admin/ui/_datatable_thead.html.erb +105 -0
  69. data/app/views/smriti/admin/ui/_details.html.erb +10 -0
  70. data/app/views/smriti/admin/ui/_flash.html.erb +6 -0
  71. data/app/views/smriti/admin/ui/_table.html.erb +8 -0
  72. data/config/importmap.rb +9 -0
  73. data/config/locales/ar.yml +223 -0
  74. data/config/locales/de.yml +230 -0
  75. data/config/locales/en-AU-ocker.yml +223 -0
  76. data/config/locales/en-AU.yml +202 -0
  77. data/config/locales/en-BORK.yml +225 -0
  78. data/config/locales/en-CA.yml +223 -0
  79. data/config/locales/en-GB.yml +223 -0
  80. data/config/locales/en-LOL.yml +219 -0
  81. data/config/locales/en-SCOT.yml +223 -0
  82. data/config/locales/en-SHAKESPEARE.yml +225 -0
  83. data/config/locales/en-US-pirate.yml +222 -0
  84. data/config/locales/en-US.yml +225 -0
  85. data/config/locales/en-YODA.yml +221 -0
  86. data/config/locales/en.yml +223 -0
  87. data/config/locales/es.yml +226 -0
  88. data/config/locales/fa.yml +223 -0
  89. data/config/locales/fr-CA.yml +227 -0
  90. data/config/locales/fr.yml +227 -0
  91. data/config/locales/he.yml +218 -0
  92. data/config/locales/hi.yml +223 -0
  93. data/config/locales/it.yml +225 -0
  94. data/config/locales/ja-JP.yml +215 -0
  95. data/config/locales/pt.yml +225 -0
  96. data/config/locales/ru.yml +228 -0
  97. data/config/locales/ur.yml +225 -0
  98. data/config/locales/zh-CN.yml +214 -0
  99. data/config/locales/zh-TW.yml +214 -0
  100. data/config/routes.rb +36 -0
  101. data/lib/ext/exception.rb +20 -0
  102. data/lib/generators/smriti/install/install_generator.rb +86 -0
  103. data/lib/generators/smriti/install/templates/create_mat_view_definitions.rb +29 -0
  104. data/lib/generators/smriti/install/templates/create_mat_view_runs.rb +32 -0
  105. data/lib/generators/smriti/install/templates/smriti_initializer.rb +23 -0
  106. data/lib/smriti/admin/auth_bridge.rb +93 -0
  107. data/lib/smriti/admin/default_auth.rb +62 -0
  108. data/lib/smriti/configuration.rb +58 -0
  109. data/lib/smriti/engine.rb +82 -0
  110. data/lib/smriti/helpers/ui_test_ids.rb +49 -0
  111. data/lib/smriti/jobs/adapter.rb +81 -0
  112. data/lib/smriti/service_response.rb +75 -0
  113. data/lib/smriti/services/base_service.rb +471 -0
  114. data/lib/smriti/services/check_matview_exists.rb +76 -0
  115. data/lib/smriti/services/concurrent_refresh.rb +94 -0
  116. data/lib/smriti/services/create_view.rb +173 -0
  117. data/lib/smriti/services/delete_view.rb +111 -0
  118. data/lib/smriti/services/regular_refresh.rb +90 -0
  119. data/lib/smriti/services/swap_refresh.rb +181 -0
  120. data/lib/smriti/version.rb +21 -0
  121. data/lib/smriti.rb +64 -0
  122. data/lib/tasks/helpers.rb +185 -0
  123. data/lib/tasks/smriti_tasks.rake +151 -0
  124. metadata +206 -0
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: 0b61c4232f0786ee6491d8bdaaf802ec9736465e492816afcf0b94e14c38301b
4
+ data.tar.gz: 789a21799a501a393f5b89c870717770592adf6a54c66b79a80c9abc253e73de
5
+ SHA512:
6
+ metadata.gz: ced5a07863c0ad29bf2c6c80f97f8b826dcbdbc5d4927a8c393c7c7200affaf5599f6a98bdf2791323261a35683c4a933835fddca9452acff05f1032f00ecb11
7
+ data.tar.gz: 6d245a12638e631a2d6e9bc9af26301b7e1f2691299223cedb76f0f2a58037111d9858ccebf322e5eb83c52832c32412533ca8245c85e1460d593a334a6a0f1a
data/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2025 The smriti Authors
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,168 @@
1
+ # smriti (engine)
2
+
3
+ [![Gem](https://img.shields.io/gem/v/smriti.svg?style=flat-square)](https://rubygems.org/gems/smriti)
4
+ [![CI](https://github.com/Code-Vedas/smriti/actions/workflows/ci.yml/badge.svg)](https://github.com/Code-Vedas/smriti/actions/workflows/ci.yml)
5
+ ![License](https://img.shields.io/badge/license-MIT-blue.svg?style=flat-square)
6
+
7
+ Rails engine that manages **PostgreSQL materialized views** with definitions, services, background jobs, and Rake tasks.
8
+
9
+ ---
10
+
11
+ ## Quickstart (diagram)
12
+
13
+ ```mermaid
14
+ flowchart LR
15
+ A[Define MatViewDefinition] --> B[Create]
16
+ B --> C[Unique Index for CONCURRENT]
17
+ C --> D[Refresh: regular or concurrent or swap]
18
+ D --> E[Read MV]
19
+ D --> F[Track runs]
20
+ ```
21
+
22
+ ---
23
+
24
+ ## Install
25
+
26
+ ```bash
27
+ bundle add smriti
28
+ bin/rails g smriti:install
29
+ bin/rails db:migrate
30
+ ```
31
+
32
+ ```ruby
33
+ # config/initializers/smriti.rb
34
+ Smriti.configure do |c|
35
+ c.job_queue = :default
36
+ end
37
+ ```
38
+
39
+ ---
40
+
41
+ ## Define a view
42
+
43
+ ```ruby
44
+ defn = Smriti::MatViewDefinition.create!(
45
+ name: 'mv_user_activity',
46
+ sql: <<~SQL,
47
+ SELECT u.id AS user_id,
48
+ COUNT(a.*) AS accounts_count,
49
+ COUNT(e.*) AS events_count,
50
+ COUNT(s.*) AS sessions_count
51
+ FROM users u
52
+ LEFT JOIN accounts a ON a.user_id = u.id
53
+ LEFT JOIN events e ON e.user_id = u.id
54
+ LEFT JOIN sessions s ON s.user_id = u.id
55
+ GROUP BY u.id
56
+ SQL
57
+ refresh_strategy: :concurrent,
58
+ unique_index_columns: ['user_id']
59
+ )
60
+ ```
61
+
62
+ ---
63
+
64
+ ## Services & Jobs
65
+
66
+ ```ruby
67
+ # Create
68
+ Smriti::Services::CreateView.new(defn, force: true).call
69
+ Smriti::CreateViewJob.perform_later(defn.id, force: true)
70
+
71
+ # Refresh
72
+ Smriti::Services::RegularRefresh.new(defn, row_count_strategy: :estimated).call
73
+ Smriti::RefreshViewJob.perform_later(defn.id, row_count_strategy: :exact)
74
+
75
+ # Delete
76
+ Smriti::Services::DeleteView.new(defn, cascade: false).call
77
+ Smriti::DeleteViewJob.perform_later(defn.id, cascade: true)
78
+ ```
79
+
80
+ **Uniform response**: `status`, `meta`, `success?` / `error?`.
81
+
82
+ ---
83
+
84
+ ## Enqueue adapter
85
+
86
+ ```ruby
87
+ Smriti::Jobs::Adapter.enqueue(job_class, queue: :default, args: [...])
88
+ ```
89
+
90
+ - Uses your configured backend; **no guessing**.
91
+ - Supports **ActiveJob**, **Sidekiq**, **Resque**.
92
+
93
+ ---
94
+
95
+ ## Rake tasks
96
+
97
+ ```bash
98
+ # Create
99
+ bundle exec rake smriti:create_by_name\[VIEW_NAME,force,row_count_strategy,--yes]
100
+ bundle exec rake smriti:create_by_id\[ID,force,row_count_strategy,--yes]
101
+ bundle exec rake smriti:create_all\[force,row_count_strategy,--yes]
102
+
103
+ # Refresh
104
+ bundle exec rake smriti:refresh_by_name\[VIEW_NAME,row_count_strategy,--yes]
105
+ bundle exec rake smriti:refresh_by_id\[ID,row_count_strategy,--yes]
106
+ bundle exec rake smriti:refresh_all\[row_count_strategy,--yes]
107
+
108
+ # Delete
109
+ bundle exec rake smriti:delete_by_name\[VIEW_NAME,cascade,row_count_strategy,--yes]
110
+ bundle exec rake smriti:delete_by_id\[ID,cascade,row_count_strategy,--yes]
111
+ bundle exec rake smriti:delete_all\[cascade,row_count_strategy,--yes]
112
+ ```
113
+
114
+ ---
115
+
116
+ ## Rails version matrix
117
+
118
+ Use the pinned Gemfiles in `gemfiles/` to exercise the engine against specific Rails releases:
119
+
120
+ ```bash
121
+ # Rails 7.2.x
122
+ bundle install --gemfile gemfiles/rails_7_2.gemfile
123
+ BUNDLE_GEMFILE=gemfiles/rails_7_2.gemfile bundle exec bin/rspec-unit
124
+ BUNDLE_GEMFILE=gemfiles/rails_7_2.gemfile bundle exec bin/rspec-e2e
125
+
126
+ # Rails 8.0.x
127
+ bundle install --gemfile gemfiles/rails_8_0.gemfile
128
+ BUNDLE_GEMFILE=gemfiles/rails_8_0.gemfile bundle exec bin/rspec-unit
129
+ BUNDLE_GEMFILE=gemfiles/rails_8_0.gemfile bundle exec bin/rspec-e2e
130
+
131
+ # Rails 8.1.x
132
+ bundle install --gemfile gemfiles/rails_8_1.gemfile
133
+ BUNDLE_GEMFILE=gemfiles/rails_8_1.gemfile bundle exec bin/rspec-unit
134
+ BUNDLE_GEMFILE=gemfiles/rails_8_1.gemfile bundle exec bin/rspec-e2e
135
+
136
+ # Rails main branch
137
+ bundle install --gemfile gemfiles/rails_edge.gemfile
138
+ BUNDLE_GEMFILE=gemfiles/rails_edge.gemfile bundle exec bin/rspec-unit
139
+ BUNDLE_GEMFILE=gemfiles/rails_edge.gemfile bundle exec bin/rspec-e2e
140
+ ```
141
+
142
+ The CI workflow consumes the same Gemfiles to run the unit suite across the compatibility matrix.
143
+
144
+ ---
145
+
146
+ ## Screenshots
147
+
148
+ The end-to-end tests generate screenshots for visual verification. They are uploaded as artifacts in the GitHub Actions workflow runs.
149
+ After creating a pull request, apply the `e2e-screenshots` label to trigger screenshot generation.
150
+
151
+ You can trigger screenshot in your local setup as well:
152
+
153
+ ```bash
154
+ # 'en' is the desired locale
155
+ SCREENSHOT_LANG=en bin/rspec-app-screenshot
156
+
157
+ # OR
158
+
159
+ # runs for all configured locales (this may take long time)
160
+ bin/rspec-app-screenshots
161
+ ```
162
+
163
+ ## Docs & policies
164
+
165
+ - Root README: [../README.md](../README.md)
166
+ - **Contributing:** [../CONTRIBUTING.md](../CONTRIBUTING.md)
167
+ - **Security policy:** [../SECURITY.md](../SECURITY.md)
168
+ - **Code of Conduct:** [../CODE_OF_CONDUCT.md](../CODE_OF_CONDUCT.md)
data/Rakefile ADDED
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Copyright Codevedas Inc. 2025-present
4
+ #
5
+ # This source code is licensed under the MIT license found in the
6
+ # LICENSE file in the root directory of this source tree.
7
+
8
+ require 'bundler/setup'
9
+
10
+ APP_RAKEFILE = File.expand_path('spec/dummy/Rakefile', __dir__)
11
+ load 'rails/tasks/engine.rake'
12
+
13
+ load 'rails/tasks/statistics.rake'
14
+
15
+ require 'bundler/gem_tasks'
@@ -0,0 +1,18 @@
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <svg width="128" height="128" viewBox="0 0 256 256" fill="none"
3
+ xmlns="http://www.w3.org/2000/svg" role="img" aria-labelledby="title desc">
4
+ <title id="title">Smriti Monogram</title>
5
+ <desc id="desc">Stylized M and V mark for Materialized Views. Transparent background.</desc>
6
+
7
+ <!-- subtle outer ring -->
8
+ <circle cx="128" cy="128" r="110" stroke="#0F172A" stroke-opacity="0.18" stroke-width="10" fill="none"/>
9
+
10
+ <!-- V (accent) -->
11
+ <path d="M64 64 L128 192 L192 64" stroke="#10B981" stroke-width="20" stroke-linecap="round" stroke-linejoin="round"/>
12
+
13
+ <!-- M (primary) -->
14
+ <path d="M48 176 L92 80 L128 144 L164 80 L208 176" stroke="#0F172A" stroke-width="20" stroke-linecap="round" stroke-linejoin="round"/>
15
+
16
+ <!-- optional tiny dot to balance negative space -->
17
+ <circle cx="128" cy="208" r="4" fill="#10B981" fill-opacity="0.9"/>
18
+ </svg>
@@ -0,0 +1,18 @@
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <svg width="128" height="128" viewBox="0 0 256 256" fill="none"
3
+ xmlns="http://www.w3.org/2000/svg" role="img" aria-labelledby="title desc">
4
+ <title id="title">Smriti Monogram</title>
5
+ <desc id="desc">Stylized M and V mark for Materialized Views. Transparent background.</desc>
6
+
7
+ <!-- subtle outer ring -->
8
+ <circle cx="128" cy="128" r="110" stroke="#0F172A" stroke-opacity="0.18" stroke-width="10" fill="none"/>
9
+
10
+ <!-- V (accent) -->
11
+ <path d="M64 64 L128 192 L192 64" stroke="#10B981" stroke-width="20" stroke-linecap="round" stroke-linejoin="round"/>
12
+
13
+ <!-- M (primary) -->
14
+ <path d="M48 176 L92 80 L128 144 L164 80 L208 176" stroke="#0F172A" stroke-width="20" stroke-linecap="round" stroke-linejoin="round"/>
15
+
16
+ <!-- optional tiny dot to balance negative space -->
17
+ <circle cx="128" cy="208" r="4" fill="#10B981" fill-opacity="0.9"/>
18
+ </svg>
@@ -0,0 +1,5 @@
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <svg viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg" fill="none">
3
+ <path d="M64 64 L128 192 L192 64" stroke="currentColor" stroke-width="28" stroke-linecap="round" stroke-linejoin="round"/>
4
+ <path d="M48 176 L92 80 L128 144 L164 80 L208 176" stroke="currentColor" stroke-width="28" stroke-linecap="round" stroke-linejoin="round"/>
5
+ </svg>