Posso obter a mesma saída da consulta SQL abaixo usando o ActiveRecord?
SELECT d.title, COUNT(ep.*)
FROM posts d
LEFT JOIN comments ep
ON d.title = ep.post_title
GROUP BY d.title;
Tentei Post.joins(:comments).group(:title).count('comments.*')
.
Posso obter a mesma saída da consulta SQL abaixo usando o ActiveRecord?
SELECT d.title, COUNT(ep.*)
FROM posts d
LEFT JOIN comments ep
ON d.title = ep.post_title
GROUP BY d.title;
Tentei Post.joins(:comments).group(:title).count('comments.*')
.
Revisito Ruby on Rails e brinco com Koans. Não entendo o porquê array[4,0]
e array[4,100]
retorno []
em vez denil
def test_slicing_arrays
array = [:peanut, :butter, :and, :jelly]
assert_equal [:peanut], array[0,1]
assert_equal [:peanut, :butter], array[0,2]
assert_equal [:and, :jelly], array[2,2]
assert_equal [:and, :jelly], array[2,20]
assert_equal [], array[4,0]
assert_equal [], array[4,100]
assert_equal nil, array[5,0]
end
Tenho o parent.yml.erb abaixo:
template:
spec:
- image:
env:
- name: env_a
valueFrom:
...
- name: env_b
valueFrom:
...
- name: env_c
valueFrom:
...
- name: env_d
valueFrom:
...
Quero extrair env_b
e env_c
colocar em um arquivo child.yml.erb separado e incorporá-lo em parent.yml.erb e tentei o seguinte.
pai.yml.erb:
template:
spec:
- image:
env:
- name: env_a
valueFrom:
...
<%= ERB.new(File.read('child.yml.erb').gsub(/^/, ' ' * 8)).result(binding) %>
- name: env_d
valueFrom:
...
criança.erb.yml:
- name: env_b
valueFrom:
...
- name: env_c
valueFrom:
...
O resultado após renderizar parent.yml.erb com ERB.new(File.read('parent.yml.erb')).result(binding)
:
- name: env_b
valueFrom:
...
- name: env_c
valueFrom:
...
- name: env_d
valueFrom:
...
env_b
, env_c
e tudo abaixo é renderizado, mas tudo acima <%= ERB.new(File.read('child.yml.erb').gsub(/^/, ' ' * 8)).result(binding) %>
é removido. Há algo errado aqui?
O .gsub(/^/, ' ' * 8)
objetivo é manter o mesmo recuo. Continuo com o mesmo problema mesmo removendo o gsub
.
Estou tentando criar um novo registro de livro pelo painel de administração ativo, mas ele não é criado ao clicar no botão "Criar". Quando clico em "Criar", há uma solicitação POST para /admin/books contendo os dados do formulário. Mas, em vez de retornar um 302 e redirecionar para a lista de livros, ele retorna um 200 e o que parece ser o formulário de atualização. Embora retorne um 200, quando vou ao console para verificar se o livro foi criado, ele não é criado. Adicionei manualmente uma ação "novo" e "criar" ao modelo e isso funciona. A ação de criação funciona bem para outros modelos e afeta apenas o meu modelo de livro.
Aqui está o código no meu arquivo admin/books.rb
ActiveAdmin.register Book do
permit_params :title, :page_count, :book_type, :year_released, :price, :genre_id, :language_id, :author_id
# Displays the drop down menus correctly as well as all other inputs in order of my choosing
form do |show|
show.inputs "Details" do
show.input :language_id, as: :select, collection: Language.all.collect { |language|
[ language.language_short, language.id ] }
show.input :author_id, as: :select, collection: Author.all.collect { |author| [ author.full_name, author.age,
author.gender_short, author.biography, author.id ] }
show.input :title
show.input :genre_id, as: :select, collection: Genre.all.collect { |genre| [ genre.genre, genre.id ] }
show.input :book_type
show.input :year_released
show.input :price
end
show.actions
end
end
Aqui está o meu modelo de livro
class Book < ApplicationRecord
# Validates that all columns are not empty
validates :title, :page_count, :book_type, :year_released, :price, presence: true
# Validates that the title is unique
validates :title, uniqueness: true
# Associations between its parent tables and child table
# has_and_belongs_to_many :genres
belongs_to :language
belongs_to :author
has_many :purchases
def self.ransackable_associations(auth_object = nil)
[ "author", "genres", "language", "purchases" ]
end
def self.ransackable_attributes(auth_object = nil)
[ "author_id", "book_type", "created_at", "genre_id", "id", "language_id", "page_count", "price", "title", "updated_at", "year_released" ]
end
end
Quando uso o formulário de livro padrão em vez do meu formulário personalizado, o atributo genre_id não aparece e continua com o mesmo comportamento (falha ao criar o livro e retorna POST 200).
Estou usando o Ruby versão 3.4.1 e o Rails versão 8.0.2.
E estes são os logs do servidor ao criar um novo livro
Started POST "/admin/books" for 127.0.0.1 at 2025-04-10 14:45:29 -0500
Processing by Admin::BooksController#create as HTML
Parameters: {"authenticity_token" => "[FILTERED]", "book" => {"language_id" => "3", "author_id" => "6", "title" => "Captain Underpants", "genre_id" => "8", "book_type" => "Physical", "year_released" => "1997", "price" => "9.99"}, "commit" => "Create Book"}
AdminUser Load (0.2ms) SELECT "admin_users".* FROM "admin_users" WHERE "admin_users"."id" = 3 ORDER BY "admin_users"."id" ASC LIMIT 1 /*action='create',application='ECommerce',controller='books'*/
TRANSACTION (0.1ms) BEGIN immediate TRANSACTION /*action='create',application='ECommerce',controller='books'*/
Book Exists? (2.9ms) SELECT 1 AS one FROM "books" WHERE "books"."title" = 'Captain Underpants' LIMIT 1 /*action='create',application='ECommerce',controller='books'*/
Language Load (0.3ms) SELECT "languages".* FROM "languages" WHERE "languages"."id" = 3 LIMIT 1 /*action='create',application='ECommerce',controller='books'*/
Author Load (0.1ms) SELECT "authors".* FROM "authors" WHERE "authors"."id" = 6 LIMIT 1 /*action='create',application='ECommerce',controller='books'*/
TRANSACTION (0.1ms) ROLLBACK TRANSACTION /*action='create',application='ECommerce',controller='books'*/
Rendering /home/segern/.rbenv/versions/3.4.1/lib/ruby/gems/3.4.0/gems/activeadmin-3.3.0/app/views/active_admin/resource/new.html.arb
Language Load (0.2ms) SELECT "languages".* FROM "languages" /*action='create',application='ECommerce',controller='books'*/
↳ app/admin/books.rb:7:in 'Enumerable#collect'
Author Load (0.7ms) SELECT "authors".* FROM "authors" /*action='create',application='ECommerce',controller='books'*/
↳ app/admin/books.rb:9:in 'Enumerable#collect'
Genre Load (0.3ms) SELECT "genres".* FROM "genres" /*action='create',application='ECommerce',controller='books'*/
↳ app/admin/books.rb:12:in 'Enumerable#collect'
Rendered /home/segern/.rbenv/versions/3.4.1/lib/ruby/gems/3.4.0/gems/activeadmin-3.3.0/app/views/active_admin/resource/new.html.arb (Duration: 57.9ms | GC: 1.0ms)
Completed 200 OK in 89ms (Views: 58.0ms | ActiveRecord: 4.5ms (7 queries, 0 cached) | GC: 1.7ms)
Started GET "/favicon.ico" for 127.0.0.1 at 2025-04-10 14:45:29 -0500
ActionController::RoutingError (No route matches [GET] "/favicon.ico"):
Espera-se que o seguinte método Ruby retorne um array ou nulo, dependendo se a chamada da API foi bem-sucedida ou não.
# @return [Array or nil] - but should not throw error if anything fails
def fetch_coords_from_api
api_url = "some_api_url"
params = { api_key: ENV["API_KEY"] }
response = ApiClient.new(api_url, params).get
return unless response
json = JSON.parse(response.body)
return unless json.is_a?(Hash) && json["result"]
addresses = json["result"]
return unless addresses.is_a?(Array) && !addresses.empty?
address = addresses.find { |add| add["zip"] == zip }
return unless address&.key?("latitude") && address&.key?("longitude")
[ address["latitude"], address["longitude"] ]
end
Algum contexto: o método faz parte de uma classe que tenta vários métodos para recuperar coordenadas (se um método falhar, passamos para o próximo).
Isso funciona como esperado, mas também parece violar diversas práticas recomendadas e convenções (em relação às múltiplas verificações de um estado inválido). Alguém pode sugerir uma maneira mais robusta/eficiente/convencional/bonita de lidar com um cenário como esse?
Recebo esse erro quando tento iniciar o servidor no modo de produção
=> Booting Puma
=> Rails 8.0.1 application starting in production
=> Run `bin/rails server --help` for more startup options
[dotenv] Loaded .env
Exiting
C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/cref.rb:63:in `const_get': uninitialized constant Openai::Openai (NameError)
@mod.const_get(@cname, false)
^^^^^^^^^^
Did you mean? OpenAI
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/cref.rb:63:in `get'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/eager_load.rb:173:in `block in actual_eager_load_dir'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/helpers.rb:47:in `block in ls'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/helpers.rb:25:in `each'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/helpers.rb:25:in `ls'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/eager_load.rb:168:in `actual_eager_load_dir'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/eager_load.rb:17:in `block (2 levels) in eager_load'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/eager_load.rb:16:in `each'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/eager_load.rb:16:in `block in eager_load'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/eager_load.rb:10:in `synchronize'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader/eager_load.rb:10:in `eager_load'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader.rb:430:in `block in eager_load_all'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader.rb:428:in `each'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/zeitwerk-2.7.2/lib/zeitwerk/loader.rb:428:in `eager_load_all'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/application/finisher.rb:79:in `block in <module:Finisher>'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/initializable.rb:32:in `instance_exec'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/initializable.rb:32:in `run'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/initializable.rb:61:in `block in run_initializers'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:231:in `block in tsort_each'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:353:in `block (2 levels) in each_strongly_connected_component'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:434:in `each_strongly_connected_component_from'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:352:in `block in each_strongly_connected_component'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:350:in `each'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:350:in `call'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:350:in `each_strongly_connected_component'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:229:in `tsort_each'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/tsort.rb:208:in `tsort_each'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/initializable.rb:60:in `run_initializers'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/application.rb:440:in `initialize!'
from D:/PROGRAMMING/RUBY/addmemore_backend/config/environment.rb:5:in `<main>'
from config.ru:3:in `require_relative'
from config.ru:3:in `block (2 levels) in <main>'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rack-3.1.8/lib/rack/builder.rb:108:in `eval'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rack-3.1.8/lib/rack/builder.rb:108:in `new_from_string'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rack-3.1.8/lib/rack/builder.rb:97:in `load_file'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rack-3.1.8/lib/rack/builder.rb:67:in `parse_file'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rackup-2.2.1/lib/rackup/server.rb:354:in `build_app_and_options_from_config'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rackup-2.2.1/lib/rackup/server.rb:263:in `app'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rackup-2.2.1/lib/rackup/server.rb:424:in `wrapped_app'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rackup-2.2.1/lib/rackup/server.rb:326:in `block in start'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rackup-2.2.1/lib/rackup/server.rb:382:in `handle_profiling'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/rackup-2.2.1/lib/rackup/server.rb:325:in `start'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/commands/server/server_command.rb:38:in `start'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/commands/server/server_command.rb:145:in `block in perform'
from <internal:kernel>:90:in `tap'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/commands/server/server_command.rb:136:in `perform'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/thor-1.3.2/lib/thor/command.rb:28:in `run'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/thor-1.3.2/lib/thor/invocation.rb:127:in `invoke_command'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/command/base.rb:178:in `invoke_command'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/thor-1.3.2/lib/thor.rb:538:in `dispatch'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/command/base.rb:73:in `perform'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/command.rb:65:in `block in invoke'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/command.rb:143:in `with_argv'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/command.rb:63:in `invoke'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/railties-8.0.1/lib/rails/commands.rb:18:in `<main>'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/bundled_gems.rb:69:in `require'
from D:/APPS/Ruby33-x64/lib/ruby/3.3.0/bundled_gems.rb:69:in `block (2 levels) in replace_require'
from C:/Users/tolia/.local/share/gem/ruby/3.3.0/gems/bootsnap-1.18.4/lib/bootsnap/load_path_cache/core_ext/kernel_require.rb:30:in `require'
from D:/PROGRAMMING/RUBY/addmemore_backend/bin/rails:4:in `<main>'
Tenho openai.rb
na app/services/openai/
pasta require "net/http" require "uri" require "json"
class OpenAI
def initialize(api_key)
@api_key = api_key
end
def generate_comments(number_of_comments, post_description, customer_query)
# some code ...
end
end
Também incluo em application_controller.rb
require_relative "../services/openai/openai"
Anteriormente, resolvi esses problemas renomeando pastas, módulos e classes, mas agora ficou muito chato, porque a classe é nomeada OpenAI
, mas o Rails tenta encontrá-laOpenai
Ao consultar uma API que responde em XML surge um elemento interessante ao rastrear no console dados fictícios, layout não :
<address>SUTTER 10821
1032 LALALAND</address>
a análise está sendo executada pelo httparty
ruby gem.
A quebra de linha na string é salva como \n
em um tipo de dado text. No entanto, ao renderizar em rails com .html_safe
, nenhuma nova linha é renderizada.
Qual é a melhor maneira de processar o atributo de endereço no momento da gravação do registro?
Estou com dificuldades para fazer um counter cache funcionar em um has_one :through e estou com um comportamento estranho. Quero ter um counter_cache no Company
modelo que armazene em cache o número de associados Location
(through the BusinessUnit
model).
Objetos:
class Company < ApplicationRecord
has_many :business_units, class_name: :BusinessUnit, dependent: :destroy
has_many :locations, through: :business_units, dependent: :destroy, counter_cache: :locations_count
end
class BusinessUnit < ApplicationRecord
belongs_to :company, counter_cache: :locations_count
has_many :locations, dependent: :destroy
end
class Location < ApplicationRecord
belongs_to :business_unit
has_one :company, through: :business_unit
end
Esquema:
ActiveRecord::Schema[7.1].define(version: 2025_04_03_203549) do
enable_extension "plpgsql"
create_table "business_units", force: :cascade do |t|
t.string "name"
t.bigint "company_id", null: false
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.index ["company_id"], name: "index_business_units_on_company_id"
end
create_table "companies", force: :cascade do |t|
t.string "name"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.integer "locations_count", default: 0, null: false
end
create_table "locations", force: :cascade do |t|
t.string "name"
t.bigint "business_unit_id", null: false
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.index ["business_unit_id"], name: "index_locations_on_business_unit_id"
end
add_foreign_key "business_units", "companies"
add_foreign_key "locations", "business_units"
end
Meu entendimento com base nessas respostas do SO ( 1 , 2 ) é que isso deve funcionar. Não está bem documentado e é um pouco contraintuitivo como o Rails sabe automaticamente se referir à contagem de locais, mas, de fato, parece ter alguma mágica do Rails e chamar Company.reset_counters(1, :locations)
corretamente define o contador para corresponder ao número de locais.
NO ENTANTO , o contador não atualiza automaticamente (para cima ou para baixo) quando locais são adicionados ou removidos, e se uma BusinessUnit com N locais for atualizada e atribuída a outra empresa, o contador só aumenta/diminui em 1 (em vez dos N locais esperados). Esse comportamento está presente no Rails 8.0.1 e 7.1.3 (que usei acidentalmente para construir meu MRE).
Há algo óbvio que estou esquecendo e que pode fazer o counter_cache funcionar corretamente?
Estou procurando criar uma barra de pesquisa que funcione em qualquer página para encontrar itens no meu banco de dados de filmes. Tenho a barra de pesquisa funcionando apenas quando já estou na página de índice de filmes, mas ela não funciona em nenhuma outra página, ela apenas recarrega a visualização atual com a pesquisa no caminho. Como posso redirecionar para a página de índice de filmes quando envio minha pesquisa?
<form class="d-flex">
<%= form_tag movies_path, method: :get do %>
<%= text_field_tag :search, params[:search], class:"form-control me-2", placeholder:"Search" %>
<%= submit_tag "Search", :title => nil, class:"btn btn-outline-primary" %>
<% end %>
</form>
Tenho um modelo Entity
que pode ter offers
e invoices
. O usuário pode criar formatos de número personalizados para exibir o número da fatura. Por exemplo, o número da fatura interna 23
pode se tornar 2025INV00000023
ou algo assim. Agora, para offers
, precisamos ser capazes de usar um formato de número diferente. 23
pode se tornar 2025OFF00000023
.
Fiz isso criando uma NumberFormat
classe onde o usuário pode configurar formatos de números de várias maneiras. Os formatos de números são anexados a entidades porque todas as faturas de uma entidade precisam ter numeração uniforme/consistente para fins fiscais/contábeis.
Meu Entity
modelo começa assim:
class Entity < ApplicationRecord
has_many :invoices, dependent: :restrict_with_error
has_many :offers, dependent: :restrict_with_error
alguns mais não relacionadoshas_many
belongs_to :invoice_number_format, class_name: 'NumberFormat'
belongs_to :offer_number_format, class_name: 'NumberFormat'
Agora minha NumberFormat
classe precisa ter a outra ponta do belongs_to
relacionamento. Mas não tenho certeza se a duplicação ali é a abordagem correta para evitar que dependent: :restrict_with_error
o usuário exclua formatos de números que ainda estão em uso. O autoformatador do VSCode apenas exclui o segundo ao salvar. Posso, é claro, salvar sem formatar (e tomar muito cuidado para não esquecer sempre que editar este arquivo) e, após testes preliminares, parece que funciona, mas estou pensando se minha abordagem talvez não seja a ideal?
has_many :entities, inverse_of: :invoice_number_format,
dependent: :restrict_with_error
# This line gets deleted on save
has_many :entities, inverse_of: :offer_number_format,
dependent: :restrict_with_error