Class: RailsAiBridge::Serializers::Providers::BaseProviderSerializer

Inherits:
Object
  • Object
show all
Defined in:
lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb

Overview

Base class for AI assistant provider serializers (Claude, Copilot, Gemini, Codex, Cursor, Windsurf). Shared compact-mode sections: header, stack, models, gems, architecture, MCP guide, commands, footer.

Defined Under Namespace

Classes: ModelEntries, NotableGemPayload

Constant Summary collapse

MAX_KEY_MODELS =

Maximum number of key models to display in compact mode Models beyond this count are truncated with an overflow hint

15
MAX_PATTERNS =

Maximum number of architectural patterns to display Limits the patterns section to prevent excessive output

8
MAX_CONFIG_FILES =

Maximum number of configuration files to list Shows only the most important config files

5

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(context, config: RailsAiBridge.configuration) ⇒ BaseProviderSerializer

Returns a new instance of BaseProviderSerializer.

Parameters:



25
26
27
28
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 25

def initialize(context, config: RailsAiBridge.configuration)
  @context = context
  @config = config
end

Instance Attribute Details

#configObject (readonly)

Returns the value of attribute config.



21
22
23
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 21

def config
  @config
end

#contextObject (readonly)

Returns the value of attribute context.



21
22
23
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 21

def context
  @context
end

Instance Method Details

#render_architectureArray<String>

Renders the architecture section with detected styles and common patterns. Caps patterns at MAX_PATTERNS entries. Returns +[]+ when conventions are absent, have an +:error+ key, or both +:architecture+ and +:patterns+ are empty.

Returns:

  • (Array<String>)

    Lines for the architecture section, or +[]+ if unavailable.



127
128
129
130
131
132
133
134
135
136
137
138
139
140
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 127

def render_architecture
  conv = context[:conventions]
  return [] unless conv.is_a?(Hash) && !conv[:error]

  arch = conv[:architecture] || []
  patterns = conv[:patterns] || []
  return [] if arch.empty? && patterns.empty?

  lines = ['## Architecture', 'Detected architectural styles and common patterns:']
  arch.each { |p| lines << "- #{p}" }
  patterns.first(MAX_PATTERNS).each { |p| lines << "- #{p}" }
  lines << ''
  lines
end

#render_commandsArray<String>

Renders the command reference section with common dev, test, lint and migration commands. The test command is resolved via ContextSummary.test_command.

Returns:

  • (Array<String>)

    Lines for the commands section.



214
215
216
217
218
219
220
221
222
223
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 214

def render_commands
  [
    '## Commands',
    '- `bin/dev` — start dev server',
    "- `#{ContextSummary.test_command(context)}` — run tests",
    '- `bundle exec rubocop` — run linter',
    '- `rails db:migrate` — run pending migrations',
    ''
  ]
end

#render_compactString

Renders the default compact AI context document (newline-joined sections). Enforces Configuration#claude_max_lines by trimming with an MCP pointer when exceeded. Subclasses may override entirely or compose with individual #render_* helpers.

Returns:

  • (String)

    Compact markdown body.



35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 35

def render_compact
  lines = []
  lines.concat(render_header)
  lines.concat(render_stack_overview)
  lines.concat(render_key_models)
  lines.concat(render_notable_gems)
  lines.concat(render_architecture)
  lines.concat(render_semantic_insights)
  lines.concat(render_key_considerations)
  lines.concat(Formatters::Providers::McpGuideFormatter.new(context).call.split("\n"))
  lines.concat(render_key_config_files)
  lines.concat(render_commands)
  lines.concat(render_footer)

  line_enforcer.enforce(lines).join("\n")
end

Renders the closing engineering rules and regeneration attribution footer. Delegates to SharedAssistantGuidance.compact_engineering_rules_footer_lines.

Returns:

  • (Array<String>)

    Lines for the footer section.



229
230
231
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 229

def render_footer
  SharedAssistantGuidance.compact_engineering_rules_footer_lines(context)
end

#render_headerArray<String>

Renders the header section of the context file.

Returns:

  • (Array<String>)

    Lines for the header.



54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 54

def render_header
  [
    "# #{context[:app_name]} — AI Context",
    '',
    "> Auto-generated by rails-ai-bridge v#{RailsAiBridge::VERSION}",
    "> Generated: #{context[:generated_at]}",
    "> Rails #{context[:rails_version]} | Ruby #{context[:ruby_version]}",
    '',
    "This file provides a high-level overview of this Rails application's",
    'structure, patterns, and conventions. As an AI assistant, use this context',
    'to quickly understand the project and generate idiomatic code that',
    'adheres to its design decisions. For deeper dives, use the live',
    'MCP tools referenced throughout this document.'
  ]
end

#render_key_config_filesArray<String>

Renders the key configuration files section. Caps display at MAX_CONFIG_FILES files. Returns +[]+ when conventions are absent, have an +:error+ key, or +:config_files+ is empty.

Returns:

  • (Array<String>)

    Lines for the key config files section, or +[]+ if unavailable.



197
198
199
200
201
202
203
204
205
206
207
208
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 197

def render_key_config_files
  conv = context[:conventions]
  return [] unless conv.is_a?(Hash) && !conv[:error]

  config_files = ContextSummary.safe_config_files(conv[:config_files], limit: MAX_CONFIG_FILES)
  return [] if config_files.empty?

  lines = ['## Key Config Files', 'Core configuration files for this application:']
  config_files.each { |f| lines << "- `#{f}`" }
  lines << ''
  lines
end

#render_key_considerationsArray<String>

Renders the static key considerations section covering performance, security, data drift, and MCP exposure. Content is fixed and does not depend on context.

Returns:

  • (Array<String>)

    Lines for the key considerations section.



179
180
181
182
183
184
185
186
187
188
189
190
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 179

def render_key_considerations
  [
    '## Key Considerations',
    '- **Performance:** For large or frequently accessed tables, always consider database performance. ' \
    'Use the `rails_get_schema` tool to verify indexes and be mindful of N+1 queries by using `includes` and other ActiveRecord optimizations.',
    '- **Security:** Treat all user-provided input as untrusted. Always use strong parameters in controllers ' \
    'and be aware of potential security vulnerabilities when using gems like `ransack` or `pg_search`.',
    '- **Data Drift:** This document is a snapshot. For the most up-to-date information, especially regarding schema and routes, use the live MCP tools.',
    '- **MCP Exposure:** The MCP tools are read-only but expose sensitive application structure. Avoid exposing the HTTP transport on untrusted networks.',
    ''
  ]
end

#render_key_modelsArray<String>

Renders the key models section, sorted by task relevance (semantic tier, complexity, route density, recent migrations, and optional DB size hints). Caps display at MAX_KEY_MODELS models and appends an overflow hint when more exist. Returns +[]+ when +context[:models]+ is nil, non-Hash, or has an +:error+ key.

Returns:

  • (Array<String>)

    Lines for the key models section, or +[]+ if unavailable.



86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 86

def render_key_models
  models = context[:models]
  return [] unless models.is_a?(Hash) && !models[:error] && models.any?

  max_show = MAX_KEY_MODELS

  lines = ['## Key Models',
           'The following are the most architecturally significant models, ordered by relevance:']
  model_entries = ModelEntries.new(models, context: context).sorted_by_relevance
  model_entries.first(max_show).each do |name, data|
    lines << model_line_formatter.format_line(name, data)
  end
  lines << "- _...#{model_entries.size - max_show} more (use `rails_get_model_details` tool)_" if model_entries.size > max_show
  lines << ''
  lines
end

#render_notable_gemsArray<String>

Renders the notable gems section grouped by category. Looks for notable gems under +:notable_gems+, +:notable+, or +:detected+ keys. Returns +[]+ when gems are absent, have an +:error+ key, or the list is empty.

Returns:

  • (Array<String>)

    Lines for the notable gems section, or +[]+ if unavailable.



108
109
110
111
112
113
114
115
116
117
118
119
120
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 108

def render_notable_gems
  notable = extract_notable_gems(context[:gems])
  return [] if notable.empty?

  lines = ['## Gems', 'Key gems, categorized by their primary function:']
  grouped = notable.group_by { |g| g[:category]&.to_s || 'other' }
  grouped.each do |category, gem_list|
    names = gem_list.map { |g| g[:name] }.join(', ')
    lines << "- **#{category}**: #{names}"
  end
  lines << ''
  lines
end

#render_semantic_insightsArray<String>

Renders semantic insights from rubydex analysis when available. Includes pattern detection results and complexity hotspot warnings. Returns +[]+ when semantic data is absent or has an +:error+ or +:info+ key.

Returns:

  • (Array<String>)

    Lines for the semantic insights section, or +[]+ if unavailable.



147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 147

def render_semantic_insights
  semantic = context[:semantic]
  return [] unless semantic.is_a?(Hash) && !semantic[:error] && !semantic[:info]

  lines = ['## Semantic Analysis (rubydex)']

  # Codebase stats
  stats = semantic[:codebase_stats]
  if stats.is_a?(Hash) && stats.any?
    lines << "- **Files:** #{stats[:total_files]}, **Classes:** #{stats[:total_classes]}, " \
             "**Modules:** #{stats[:total_modules]}, **Methods:** #{stats[:total_methods]}"
  end

  # Detected patterns
  patterns = semantic.dig(:patterns, :common_patterns)
  lines << "- **Patterns:** #{patterns.join(', ')}" if patterns.is_a?(Array) && patterns.any?

  # Complexity hotspots
  hotspots = semantic[:complexity_hotspots]
  if hotspots.is_a?(Array) && hotspots.any?
    hotspot_list = hotspots.first(5).map { |h| "#{h[:name]} (score: #{h[:complexity_score]})" }.join(', ')
    lines << "- **Complexity hotspots:** #{hotspot_list}"
  end

  lines << ''
  lines
end

#render_stack_overviewArray<String>

Renders the stack overview section. Includes database adapter/table count, model count, routes, auth gems, async jobs/mailers/channels, and pending migrations when available. Silently skips any sub-section whose context key is missing or has an +:error+ key.

Returns:

  • (Array<String>)

    Lines for the stack overview section, always non-empty.



76
77
78
# File 'lib/rails_ai_bridge/serializers/providers/base_provider_serializer.rb', line 76

def render_stack_overview
  stack_overview_builder.build
end