graphql 2.4.15 → 2.4.16

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 (33) hide show
  1. checksums.yaml +4 -4
  2. data/lib/graphql/dashboard/detailed_traces.rb +47 -0
  3. data/lib/graphql/dashboard/installable.rb +22 -0
  4. data/lib/graphql/dashboard/limiters.rb +93 -0
  5. data/lib/graphql/dashboard/operation_store.rb +199 -0
  6. data/lib/graphql/dashboard/statics/charts.min.css +1 -0
  7. data/lib/graphql/dashboard/statics/dashboard.css +27 -0
  8. data/lib/graphql/dashboard/statics/dashboard.js +74 -9
  9. data/lib/graphql/dashboard/subscriptions.rb +96 -0
  10. data/lib/graphql/dashboard/views/graphql/dashboard/detailed_traces/traces/index.html.erb +45 -0
  11. data/lib/graphql/dashboard/views/graphql/dashboard/limiters/limiters/show.html.erb +62 -0
  12. data/lib/graphql/dashboard/views/graphql/dashboard/not_installed.html.erb +18 -0
  13. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/clients/_form.html.erb +23 -0
  14. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/clients/edit.html.erb +21 -0
  15. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/clients/index.html.erb +69 -0
  16. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/clients/new.html.erb +7 -0
  17. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/index_entries/index.html.erb +39 -0
  18. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/index_entries/show.html.erb +32 -0
  19. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/operations/index.html.erb +81 -0
  20. data/lib/graphql/dashboard/views/graphql/dashboard/operation_store/operations/show.html.erb +71 -0
  21. data/lib/graphql/dashboard/views/graphql/dashboard/subscriptions/subscriptions/show.html.erb +41 -0
  22. data/lib/graphql/dashboard/views/graphql/dashboard/subscriptions/topics/index.html.erb +55 -0
  23. data/lib/graphql/dashboard/views/graphql/dashboard/subscriptions/topics/show.html.erb +40 -0
  24. data/lib/graphql/dashboard/views/layouts/graphql/dashboard/application.html.erb +49 -1
  25. data/lib/graphql/dashboard.rb +45 -29
  26. data/lib/graphql/execution/interpreter.rb +3 -2
  27. data/lib/graphql/execution/multiplex.rb +1 -1
  28. data/lib/graphql/language/parser.rb +13 -6
  29. data/lib/graphql/query.rb +2 -1
  30. data/lib/graphql/tracing/perfetto_trace.rb +4 -4
  31. data/lib/graphql/version.rb +1 -1
  32. metadata +22 -3
  33. data/lib/graphql/dashboard/views/graphql/dashboard/traces/index.html.erb +0 -63
@@ -52,18 +52,77 @@ async function openOnPerfetto(operationName, tracePath) {
52
52
  }, 100)
53
53
  }
54
54
 
55
- async function deleteTrace(tracePath, event) {
55
+ function getCsrfToken() {
56
+ return document.querySelector("meta[name='csrf-token']").content
57
+ }
58
+
59
+ function deleteTrace(tracePath) {
56
60
  if (confirm("Are you sure you want to permanently delete this trace?")) {
57
- var response = await fetch(tracePath, { method: "DELETE", headers: {
58
- "X-CSRF-Token": document.querySelector("meta[name='csrf-token']").content
59
- } })
60
- if (response.ok) {
61
- var row = event.target.closest("tr")
62
- row.remove()
63
- } else {
64
- console.error("Delete request failed for", tracePath, response)
61
+ fetch(tracePath, { method: "DELETE", headers: {
62
+ "X-CSRF-Token": getCsrfToken()
63
+ } }).then(function(_response) {
64
+ window.location.reload()
65
+ })
66
+ }
67
+ }
68
+
69
+ function deleteAllTraces(path) {
70
+ if (confirm("Are you sure you want to permanently delete ALL traces?")) {
71
+ fetch(path, { method: "DELETE", headers: {
72
+ "X-CSRF-Token": getCsrfToken()
73
+ } }).then(function(_response) {
74
+ window.location.reload()
75
+ })
76
+ }
77
+ }
78
+
79
+ function deleteAllSubscriptions(path) {
80
+ if (confirm("This will:\n\n- Remove all subscriptions from the database\n- Stop updates to all current subscribers\n\nAre you sure?")) {
81
+ fetch(path, { method: "POST", headers: {
82
+ "X-CSRF-Token": getCsrfToken()
83
+ } }).then(function(_response) {
84
+ window.location.reload()
85
+ })
86
+ }
87
+ }
88
+
89
+ function sendArchive(clientName) {
90
+ var values = []
91
+ document.querySelectorAll(".archive-check:checked").forEach(function(el) {
92
+ values.push(el.value)
93
+ })
94
+ if (values.length == 0) {
95
+ return
96
+ }
97
+ var mode = window.location.pathname.includes("/archived") ? "/unarchive" : "/archive"
98
+ if (mode == "/archive") {
99
+ if (!confirm("Are you sure you want to archive these operations? They won't be usable by clients while archived.")) {
100
+ return
101
+ }
102
+ } else {
103
+ if (!confirm("Are you sure you want to reactivate these operations? They'll be available to clients again.")) {
104
+ return
105
+ }
106
+ }
107
+ var url = window.location.pathname.replace("/archived", "")
108
+ url += mode
109
+ var data
110
+
111
+ if (clientName) {
112
+ data = {
113
+ operation_aliases: values
114
+ }
115
+ } else {
116
+ data = {
117
+ digests: values
65
118
  }
66
119
  }
120
+ fetch(url, { method: "POST", body: JSON.stringify(data), headers: {
121
+ "X-CSRF-Token": getCsrfToken(),
122
+ "Content-Type": "application/json",
123
+ }}).then(function(_response) {
124
+ window.location.reload()
125
+ })
67
126
  }
68
127
 
69
128
  document.addEventListener("click", function(event) {
@@ -72,7 +131,13 @@ document.addEventListener("click", function(event) {
72
131
  openOnPerfetto(dataset.perfettoOpen, dataset.perfettoPath)
73
132
  } else if (dataset.perfettoDelete) {
74
133
  deleteTrace(dataset.perfettoDelete, event)
134
+ } else if (dataset.perfettoDeleteAll) {
135
+ deleteAllTraces(dataset.perfettoDeleteAll)
136
+ } else if (dataset.subscriptionsDeleteAll) {
137
+ deleteAllSubscriptions(dataset.subscriptionsDeleteAll)
75
138
  } else if (event.target.id == "themeToggle") {
76
139
  toggleTheme()
140
+ } else if (dataset.archiveClient || dataset.archiveAll) {
141
+ sendArchive(dataset.archiveClient)
77
142
  }
78
143
  })
@@ -0,0 +1,96 @@
1
+ # frozen_string_literal: true
2
+ module Graphql
3
+ class Dashboard < Rails::Engine
4
+ module Subscriptions
5
+ class BaseController < Graphql::Dashboard::ApplicationController
6
+ include Installable
7
+
8
+ def feature_installed?
9
+ schema_class.subscriptions.is_a?(GraphQL::Pro::Subscriptions)
10
+ end
11
+
12
+ INSTALLABLE_COMPONENT_HEADER_HTML = "GraphQL-Pro Subscriptions aren't installed on this schema yet.".html_safe
13
+ INSTALLABLE_COMPONENT_MESSAGE_HTML = <<-HTML.html_safe
14
+ Deliver live updates over
15
+ <a href="https://graphql-ruby.org/subscriptions/pusher_implementation.html">Pusher</a> or
16
+ <a href="https://graphql-ruby.org/subscriptions/ably_implementation.html"> Ably</a>
17
+ with GraphQL-Pro's subscription integrations.
18
+ HTML
19
+ end
20
+
21
+ class TopicsController < BaseController
22
+ def show
23
+ topic_name = params[:name]
24
+ all_subscription_ids = []
25
+ schema_class.subscriptions.each_subscription_id(topic_name) do |sid|
26
+ all_subscription_ids << sid
27
+ end
28
+
29
+ page = params[:page]&.to_i || 1
30
+ limit = params[:per_page]&.to_i || 20
31
+ offset = limit * (page - 1)
32
+ subscription_ids = all_subscription_ids[offset, limit]
33
+ subs = schema_class.subscriptions.read_subscriptions(subscription_ids)
34
+ show_broadcast_subscribers_count = schema_class.subscriptions.show_broadcast_subscribers_count?
35
+ subs.each do |sub|
36
+ sub[:is_broadcast] = is_broadcast = schema_class.subscriptions.broadcast_subscription_id?(sub[:id])
37
+ if is_broadcast && show_broadcast_subscribers_count
38
+ sub[:subscribers_count] = sub_count =schema_class.subscriptions.count_broadcast_subscribed(sub[:id])
39
+ sub[:still_subscribed] = sub_count > 0
40
+ else
41
+ sub[:still_subscribed] = schema_class.subscriptions.still_subscribed?(sub[:id])
42
+ sub[:subscribers_count] = nil
43
+ end
44
+ end
45
+
46
+ @topic_last_triggered_at = schema_class.subscriptions.topic_last_triggered_at(topic_name)
47
+ @subscriptions = subs
48
+ @subscriptions_count = all_subscription_ids.size
49
+ @show_broadcast_subscribers_count = show_broadcast_subscribers_count
50
+ @has_next_page = all_subscription_ids.size > offset + limit ? page + 1 : false
51
+ end
52
+
53
+ def index
54
+ page = params[:page]&.to_i || 1
55
+ per_page = params[:per_page]&.to_i || 20
56
+ offset = per_page * (page - 1)
57
+ limit = per_page
58
+ topics, all_topics_count, has_next_page = schema_class.subscriptions.topics(offset: offset, limit: limit)
59
+
60
+ @topics = topics
61
+ @all_topics_count = all_topics_count
62
+ @has_next_page = has_next_page
63
+ @page = page
64
+ end
65
+ end
66
+
67
+ class SubscriptionsController < BaseController
68
+ def show
69
+ subscription_id = params[:id]
70
+ subscriptions = schema_class.subscriptions
71
+ query_data = subscriptions.read_subscription(subscription_id)
72
+ is_broadcast = subscriptions.broadcast_subscription_id?(subscription_id)
73
+
74
+ if is_broadcast && subscriptions.show_broadcast_subscribers_count?
75
+ subscribers_count = subscriptions.count_broadcast_subscribed(subscription_id)
76
+ is_still_subscribed = subscribers_count > 0
77
+ else
78
+ subscribers_count = nil
79
+ is_still_subscribed = subscriptions.still_subscribed?(subscription_id)
80
+ end
81
+
82
+ @query_data = query_data
83
+ @still_subscribed = is_still_subscribed
84
+ @is_broadcast = is_broadcast
85
+ @subscribers_count = subscribers_count
86
+ end
87
+
88
+ def clear_all
89
+ schema_class.subscriptions.clear
90
+ flash[:success] = "All subscription data cleared."
91
+ head :no_content
92
+ end
93
+ end
94
+ end
95
+ end
96
+ end
@@ -0,0 +1,45 @@
1
+ <% content_for(:title, "Profiles") %>
2
+ <div class="row justify-content-between mt-3">
3
+ <div class="col-auto">
4
+ <h3>Detailed Profiles</h3>
5
+ </div>
6
+ <div class="col-auto">
7
+ <%= button_tag "Delete All Traces", class: "btn btn-sm btn-outline-danger", data: { perfetto_delete_all: graphql_dashboard.delete_all_detailed_traces_traces_path } %>
8
+ </div>
9
+ </div>
10
+
11
+ <div class="row">
12
+ <div class="col">
13
+ <table class="table table-striped">
14
+ <thead>
15
+ <tr>
16
+ <th>Operation</th>
17
+ <th>Duration (ms) </th>
18
+ <th>Timestamp</th>
19
+ <th>Open in Perfetto UI</th>
20
+ </tr>
21
+ </thead>
22
+ <tbody>
23
+ <% if @traces.empty? %>
24
+ <tr>
25
+ <td colspan="4" class="text-center">
26
+ <em>No traces saved yet. Read about saving traces <%= link_to "in the docs", "https://graphql-ruby.org/queries/tracing#detailed-profiles" %>.</em>
27
+ </td>
28
+ </tr>
29
+ <% end %>
30
+ <% @traces.each do |trace| %>
31
+ <tr>
32
+ <td><%= trace.operation_name %></td>
33
+ <td><%= trace.duration_ms.round(2) %></td>
34
+ <td><%= Time.at(trace.begin_ms / 1000.0).strftime("%Y-%m-%d %H:%M:%S.%L") %></td>
35
+ <td><%= link_to "View ↗", "#", data: { perfetto_open: trace.operation_name, perfetto_path: graphql_dashboard.detailed_traces_trace_path(trace.id) } %></td>
36
+ <td><%= link_to "Delete", "#", data: { perfetto_delete: graphql_dashboard.detailed_traces_trace_path(trace.id) }, class: "text-danger" %></td>
37
+ </tr>
38
+ <% end %>
39
+ </tbody>
40
+ </table>
41
+ <% if @last && @traces.size >= @last %>
42
+ <%= link_to("Previous >", graphql_dashboard.detailed_traces_traces_path(last: @last, before: @traces.last.begin_ms), class: "btn btn-outline-primary") %>
43
+ <% end %>
44
+ </div>
45
+ </div>
@@ -0,0 +1,62 @@
1
+ <% content_for(:title, @title) %>
2
+ <% if @install_path %>
3
+ <div class="row mt-3">
4
+ <div class="col">
5
+ <h3><%= @title %></h3>
6
+ <p>It looks like this limiter isn't installed yet. <a href="<%= @install_path %>">Install it now</a>.</p>
7
+ </div>
8
+ </div>
9
+ <% else %>
10
+ <div class="row mt-3 justify-content-between">
11
+ <div class="col-auto">
12
+ <h3><%= @title %></h3>
13
+ </div>
14
+ <div class="col-auto">
15
+ <div class="btn-group">
16
+ <%= link_to("This Hour", graphql_dashboard.limiters_limiter_path(params[:name], chart: "hour"), class: "btn btn-sm btn-outline-primary #{@chart_mode == "hour" ? "active" : "inactive"}", params: { chart: "hour" }) %>
17
+ <%= link_to("Today", graphql_dashboard.limiters_limiter_path(params[:name], chart: "day"), class: "btn btn-sm btn-outline-primary #{@chart_mode == "day" ? "active" : "inactive"}", params: { chart: "day" }) %>
18
+ <%= link_to("This Month", graphql_dashboard.limiters_limiter_path(params[:name], chart: "month"), class: "btn btn-sm btn-outline-primary #{@chart_mode == "month" ? "active" : "inactive"}", params: { chart: "month" }) %>
19
+ </div>
20
+ </div>
21
+ <div class="col-auto">
22
+ <%= form_tag graphql_dashboard.limiters_limiter_path(params[:name], chart: @chart_mode), method: "patch" do %>
23
+ <%= submit_tag "#{@current_soft ? "Disable" : "Enable"} Soft Limiting", class: "btn btn-sm btn-outline-warning" %>
24
+ <% end %>
25
+ </div>
26
+ </div>
27
+
28
+ <div class="row">
29
+ <div class="col ms-4">
30
+ <div id="limiter-histogram">
31
+ <table class="charts-css column multiple stacked show-labels hide-data show-primary-axis">
32
+ <thead>
33
+ <th>Date</th>
34
+ <th scope="col">Limited Requests</th>
35
+ <th scope="col">Unlimited Requests</th>
36
+ </thead>
37
+ <tbody>
38
+ <% @histogram.columns.each_with_index do |col, col_idx| %>
39
+ <tr>
40
+ <th scope="row" class="text-end"><%= col.label %></th>
41
+ <% col.values.each_with_index do |value, val_idx| %>
42
+ <td id="data-<%= col_idx %>-<%= val_idx %>">
43
+ <span class="data"><%= value.formatted_value %></span>
44
+ <span class="tooltip"><%= value.label %>: <%= value.formatted_value %><br><%= col.label %></span>
45
+ </td>
46
+ <% end %>
47
+ </tr>
48
+ <% end %>
49
+ </tbody>
50
+ </table>
51
+ </div>
52
+ </div>
53
+ </div>
54
+ <%= content_tag "style", nonce: @csp_nonce do %>
55
+ <% @histogram.columns.each_with_index do |col, col_idx| %>
56
+ <% col_max = @histogram.max_column_value.to_f %>
57
+ <% col.values.each_with_index do |val, val_idx| %>
58
+ #data-<%= col_idx %>-<%= val_idx %> { --size: <%= val.value / col_max %>}
59
+ <% end %>
60
+ <% end %>
61
+ <% end %>
62
+ <% end %>
@@ -0,0 +1,18 @@
1
+ <% content_for(:title, "Operation Store") %>
2
+
3
+ <div class="row">
4
+ <div class="col-md col-lg-8 mx-auto pt-4">
5
+ <div class="card mt-4">
6
+ <div class="card-body">
7
+ <div class="card-title">
8
+ <h2>
9
+ <%= @component_header_html %>
10
+ </h2>
11
+ </div>
12
+ <p class="card-text">
13
+ <%= @component_message_html %>
14
+ </p>
15
+ </div>
16
+ </div>
17
+ </div>
18
+ </div>
@@ -0,0 +1,23 @@
1
+ <%= form_tag((@client.persisted? ? graphql_dashboard.operation_store_client_path(name: @client.name) : graphql_dashboard.operation_store_clients_path), method: (@client.persisted? ? "patch" : "post")) do %>
2
+ <div class="row">
3
+ <label class="col-2 col-form-label">Name</label>
4
+ <div class="col">
5
+ <%= text_field_tag "client[name]", @client.name, class: "form-control", disabled: @client.persisted? %>
6
+ <div class="form-text">a unique identifier for this owner of persisted operations</div>
7
+ </div>
8
+ </div>
9
+ <div class="row">
10
+ <label class="col-2 col-form-label">Secret</label>
11
+ <div class="col">
12
+ <%= textarea_tag "client[secret]", @client.secret, class: "form-control" %>
13
+ <div class="form-text">authentication credential for <code>sync</code> transactions</div>
14
+ </div>
15
+ </div>
16
+ <div class="row">
17
+ <div class="col-auto">
18
+ <%= submit_tag "Save", class: "btn btn-outline-primary" %>
19
+ </div>
20
+ <div class="col-auto">
21
+ <%= link_to "Back", graphql_dashboard.operation_store_clients_path, class: "btn btn-outline-secondary" %>
22
+ </div>
23
+ <% end %>
@@ -0,0 +1,21 @@
1
+ <% content_for(:title, "Edit #{@client.name}") %>
2
+ <div class="row">
3
+ <div class="col">
4
+ <h1>Edit <%= @client.name %></h1>
5
+ </div>
6
+ <div>
7
+ <%= render partial: "graphql/dashboard/operation_store/clients/form" %>
8
+
9
+ <hr class="mt-5"/>
10
+ <div class="row mt-5">
11
+ <div class="col">
12
+ <div class="alert alert-danger">
13
+ <h4>Delete <%= @client.name %></h4>
14
+ <p>If you delete this client, it will no longer be able to use stored operations.</p>
15
+ <p>There is no way to undo this action.</p>
16
+ <%= form_tag(graphql_dashboard.operation_store_client_path(name: @client.name), method: "delete") do %>
17
+ <%= submit_tag "Permanently Delete #{@client.name.inspect}", class: "btn btn-outline-danger" %>
18
+ <% end %>
19
+ </div>
20
+ </div>
21
+ </div>
@@ -0,0 +1,69 @@
1
+ <% content_for(:title, "Clients") %>
2
+ <div class="row mt-3 justify-content-between">
3
+ <div class="col-4">
4
+ <h3>
5
+ <%= pluralize(@clients_page.total_count, "Client") %>
6
+ </h3>
7
+ </div>
8
+ <div class="col-auto">
9
+ <%= link_to("New Client", graphql_dashboard.new_operation_store_client_path, class: "btn btn-outline-primary") %>
10
+ </div>
11
+ </div>
12
+
13
+ <table class="table table-striped">
14
+ <thead>
15
+ <tr>
16
+ <th><%= link_to("Name", graphql_dashboard.operation_store_clients_path, params: { order_by: "name", order_dir: ((@order_by == "name" && @order_dir != :desc) ? "desc" : "asc" )}) %></th>
17
+ <th>Operations</th>
18
+ <th>Created At</th>
19
+ <th>Last Updated</th>
20
+ <th><%= link_to("Last Used At", graphql_dashboard.operation_store_clients_path, params: { order_by: "last_used_at", order_dir: ((@order_by == "last_used_at" && @order_dir != :desc) ? "desc": "asc")}) %></th>
21
+ </tr>
22
+ </thead>
23
+ <tbody>
24
+ <% if @clients_page.total_count == 0 %>
25
+ <tr>
26
+ <td colspan="5" class="text-center">
27
+ <em>To get started, create a <%= link_to "new client", graphql_dashboard.new_operation_store_client_path %>, then <%= link_to "sync operations", "https://graphql-ruby.org/operation_store/client_workflow.html" %> to your schema.</em>
28
+ </td>
29
+ </tr>
30
+ <% else %>
31
+ <% @clients_page.items.each do |client| %>
32
+ <tr>
33
+ <td><%= link_to(client.name, graphql_dashboard.edit_operation_store_client_path(name: client.name)) %></td>
34
+ <td>
35
+ <%= link_to(graphql_dashboard.operation_store_client_operations_path(client_name: client.name)) do %>
36
+ <%= client.operations_count %><% if client.archived_operations_count > 0 %> <span class="muted">(<%=client.archived_operations_count%> archived)</span><% end %>
37
+ <% end %>
38
+ </td>
39
+ <td><%= client.created_at %></td>
40
+ <td>
41
+ <% if client.operations_count == 0 %>
42
+ &mdash;
43
+ <% else %>
44
+ <%= client.last_synced_at %>
45
+ <% end %>
46
+ </td>
47
+ <td><%= client.last_used_at || "—" %></td>
48
+ </tr>
49
+ <% end %>
50
+ <% end %>
51
+ </tbody>
52
+ </table>
53
+
54
+ <div class="row">
55
+ <div class="col-auto">
56
+ <% if @clients_page.prev_page %>
57
+ <%= link_to("« prev", graphql_dashboard.operation_store_clients_path(per_page: params[:per_page], page: @clients_page.prev_page), class: "btn btn-outline-secondary") %>
58
+ <% else %>
59
+ <button class="btn btn-outline-secondary" disabled>« prev</button>
60
+ <% end %>
61
+ </div>
62
+ <div class="col-auto">
63
+ <% if @clients_page.next_page %>
64
+ <%= link_to("next »", graphql_dashboard.operation_store_clients_path(per_page: params[:per_page], page: @clients_page.next_page), class: "btn btn-outline-secondary") %>
65
+ <% else %>
66
+ <button class="btn btn-outline-secondary" disabled>next »</button>
67
+ <% end %>
68
+ </div>
69
+ </div>
@@ -0,0 +1,7 @@
1
+ <% content_for(:title, "New Client") %>
2
+ <div class="row">
3
+ <div class="col">
4
+ <h1>New Client</h1>
5
+ </div>
6
+ <div>
7
+ <%= render partial: "graphql/dashboard/operation_store/clients/form" %>
@@ -0,0 +1,39 @@
1
+ <% content_for(:title, "Index#{@search_term ? " - #{@search_term}" : ""}") %>
2
+ <div class="row mt-2">
3
+ <h3>Schema Index</h3>
4
+ <div class="col">
5
+ <p style="margin-left: 15px;">
6
+ <%= pluralize(@index_entries_page.total_count, @search_term ? "result" : "entry") %>
7
+ </p>
8
+ </div>
9
+ <div class="col">
10
+ <form method="GET" action="<%= graphql_dashboard.operation_store_index_entries_path %>" style="margin-left: auto; margin-top:-5px;">
11
+ <div class="input-group">
12
+ <%= text_field_tag "q", @search_term, class: "form-control", placeholder: "Find types, fields, arguments, or enum values" %>
13
+ <input type="submit" value="Search" class="btn btn-outline-primary btn-sm"/>
14
+ </div>
15
+ </form>
16
+ </div>
17
+ </div>
18
+ <table class="table">
19
+ <thead>
20
+ <tr>
21
+ <th>Name</th>
22
+ <th># Usages</th>
23
+ <th>Last Used At</th>
24
+ </tr>
25
+ </thead>
26
+ <tbody>
27
+ <% @index_entries_page.items.each do |entry| %>
28
+ <tr>
29
+ <td><%= link_to(entry.name, graphql_dashboard.operation_store_index_entry_path(name: entry.name)) %></td>
30
+ <td><%= entry.references_count %><% if entry.archived_references_count.nil? %><span class="muted">(missing data - call `YourSchema.operation_store.reindex` to repair index)</span><% elsif entry.archived_references_count > 0 %> <span class="muted">(<%= entry.archived_references_count %> archived)</span><% end %></td>
31
+ <td><%= entry.last_used_at %></td>
32
+ </tr>
33
+ <% end %>
34
+ </tbody>
35
+ </table>
36
+
37
+ <%=
38
+ # render_partial("_pagination")
39
+ %>
@@ -0,0 +1,32 @@
1
+ <% name = @chain.pop %>
2
+ <% content_for(:title, "Index - #{@entry.name}") %>
3
+ <div class="row mt-2">
4
+ <div class="col">
5
+ <%= link_to("Index", graphql_dashboard.operation_store_index_entries_path) %>
6
+ <% @chain.each do |c| %>
7
+ > <%= link_to(c.split(".").last, graphql_dashboard.operation_store_index_entry_path(name: c)) %>
8
+ <% end %>
9
+ > <%= name.split(".").last %>
10
+ </div>
11
+ </div>
12
+ <div class="row mt-2">
13
+ <div class="col">
14
+ <h3><%= name %></h3>
15
+ <p>
16
+ Used By:
17
+ <% if @operations.any? %>
18
+ <ul>
19
+ <% @operations.each do |operation| %>
20
+ <li>
21
+ <%= link_to(operation.name, graphql_dashboard.operation_store_operation_path(digest: operation.digest)) %><% if operation.is_archived %> <span class="muted">(archived)</span><% end %>
22
+ </li>
23
+ <% end %>
24
+ </ul>
25
+ <% else %>
26
+ <i>none</i>
27
+ <% end %>
28
+ </p>
29
+
30
+ <p>Last used at: <%= @entry.last_used_at || "—" %></p>
31
+ </div>
32
+ </div>
@@ -0,0 +1,81 @@
1
+ <div class="row mt-2">
2
+ <% if @client_operations %>
3
+ <%= content_for(:title, "#{params[:client_name]} Operations") %>
4
+ <div class="col">
5
+ <h3><%= params[:client_name] %></h3>
6
+ <ul class="nav nav-tabs">
7
+ <li class="nav-item">
8
+ <%= link_to "#{@unarchived_operations_count} Active", graphql_dashboard.operation_store_client_operations_path(client_name: params[:client_name]), class: "nav-link #{@is_archived ? "" : "active"}" %>
9
+ </li>
10
+ <li class="nav-item">
11
+ <%= link_to "#{@archived_operations_count} Archived", graphql_dashboard.archived_operation_store_client_operations_path(client_name: params[:client_name]), class: "nav-link #{@is_archived ? "active" : ""}" %>
12
+ </li>
13
+ </ul>
14
+ </div>
15
+ <% else %>
16
+ <%= content_for(:title, "Operations") %>
17
+ <div class="col">
18
+ <ul class="nav nav-tabs">
19
+ <li class="nav-item">
20
+ <%= link_to "#{@unarchived_operations_count} Active", graphql_dashboard.operation_store_operations_path, class: "nav-link #{@is_archived ? "" : "active"}" %>
21
+ </li>
22
+ <li class="nav-item">
23
+ <%= link_to "#{@archived_operations_count} Archived", graphql_dashboard.archived_operation_store_operations_path, class: "nav-link #{@is_archived ? "active" : ""}" %>
24
+ </li>
25
+ </ul>
26
+ </div>
27
+ <% end %>
28
+ </div>
29
+
30
+ <div class="row">
31
+ <div class="col">
32
+ <table class="table table-striped">
33
+ <thead>
34
+ <tr>
35
+ <th><%= link_to "Name", graphql_dashboard.operation_store_operations_path({ order_by: "name", order_dir: params[:order_dir] == "asc" ? "desc" : "asc" }) %></th>
36
+ <% if @client_operations %>
37
+ <th>Alias</th>
38
+ <% else %>
39
+ <th># Clients</th>
40
+ <% end %>
41
+ <th>Digest</th>
42
+ <th><%= link_to "Last Used At", graphql_dashboard.operation_store_operations_path({ order_by: "last_used_at", order_dir: params[:order_dir] == "asc" ? "desc" : "asc" }) %></th>
43
+ <th>
44
+ <button class="btn btn-sm btn-outline-primary" data-archive-client="<%= params[:client_name] %>" data-archive-all="<%= params[:client_name] ? nil : "true" %>">
45
+ <%= @is_archived ? "Unarchive" : "Archive" %>
46
+ </button>
47
+ </th>
48
+ </tr>
49
+ </thead>
50
+ <tbody>
51
+ <% if @operations_page.total_count == 0 %>
52
+ <tr>
53
+ <td colspan="5" class="text-center">
54
+ <% if @is_archived %>
55
+ <em><%= link_to "Archived operations", "https://graphql-ruby.org/operation_store/server_management.html#archiving-and-deleting-data" %> will appear here.</em>
56
+ <% else %>
57
+ <em>Add your first stored operations with <%= link_to "sync", "https://graphql-ruby.org/operation_store/client_workflow.html" %>.</em>
58
+ <% end %>
59
+ </td>
60
+ </tr>
61
+ <% else %>
62
+ <% @operations_page.items.each do |operation| %>
63
+ <tr>
64
+ <td><%= link_to(operation.name, graphql_dashboard.operation_store_operation_path(digest: operation.digest)) %></td>
65
+ <% if @client_operations %>
66
+ <td><code><%= operation.operation_alias %></code></td>
67
+ <% else %>
68
+ <td><%= operation.clients_count %></td>
69
+ <% end %>
70
+ <td><code><%= operation.digest %></code></td>
71
+ <td><%= operation.last_used_at %></td>
72
+ <td>
73
+ <%= check_box_tag("value", (@client_operations ? operation.operation_alias : operation.digest), class: "archive-check form-check-input") %>
74
+ </td>
75
+ </tr>
76
+ <% end %>
77
+ <% end %>
78
+ </tbody>
79
+ </table>
80
+ </div>
81
+ </div>