I've got 3 classes. Country, City, and Activity. A country has_many cities and a city has_many activities.
Some of the activities aren't ready yet, so the activities have an approval status. I'd like to set it up scopes, so a city without approved activities isn't valid and a country without valid cities is also invalid.
class ApprovalStatus < ActiveRecord::Base
def self.approved_status_id
ApprovalStatus.find_or_create_by(name: 'Approved').id
end
class Activity < ActiveRecord::Base
# Named scopes
scope :that_is_approved, -> { where approval_status_id: ApprovalStatus.approved_status_id }
class City < ActiveRecord::Base
# Named scopes
scope :that_has_valid_activities, -> { joins(:activities).merge(Activity.that_is_approved).uniq }
class Country < ActiveRecord::Base
# Named scopes
scope :that_has_valid_cities, -> { joins(:cities).uniq }
# Associations
has_many :cities, -> { that_has_valid_activities }, dependent: :destroy
I want the scope and the association as I want Country.that_has_valid_cities to only return the valid countries and only the valid cities to be shown within those valid countries when the active model serializer grabs the cities.
In the rails console: City.that_has_valid _activities runs fine.
ApprovalStatus Load (13.4ms) SELECT "approval_statuses".* FROM "approval_statuses" WHERE "approval_statuses"."name" = $1 LIMIT 1 [["name", "Approved"]]
City Load (21.6ms) SELECT DISTINCT "cities".* FROM "cities" INNER JOIN "activities" ON "activities"."city_id" = "cities"."id" WHERE "activities"."approval_status_id" = $1 [["approval_status_id", 2]]
Country.that_has_valid_cities breaks.
ApprovalStatus Load (0.4ms) SELECT "approval_statuses".* FROM "approval_statuses" WHERE "approval_statuses"."name" = $1 LIMIT 1 [["name", "Approved"]]
ActiveRecord::StatementInvalid: PG::UndefinedTable: ERROR: missing FROM-clause entry for table "activities" LINE 1: ..." ON "cities"."country_id" = "countries"."id" AND "activitie...
^
: SELECT DISTINCT "countries".* FROM "countries" INNER JOIN "cities" ON "cities"."country_id" = "countries"."id" AND "activities"."approval_status_id" = $1
It looks like it's trying to do an AND when I want it to do a second INNER JOIN.
class Country < ActiveRecord::Base
# Named scopes
scope :that_has_valid_cities, -> { joins(:cities).merge(City.that_has_valid_activities).uniq }
# Associations
has_many :cities, dependent: :destroy
Works and produces the queries:
ApprovalStatus Load (0.6ms) SELECT "approval_statuses".* FROM "approval_statuses" WHERE "approval_statuses"."name" = $1 LIMIT 1 [["name", "Approved"]]
Country Load (1.2ms) SELECT DISTINCT "countries".* FROM "countries" INNER JOIN "cities" ON "cities"."country_id" = "countries"."id" LEFT OUTER JOIN "activities" ON "activities"."city_id" = "cities"."id" WHERE "activities"."approval_status_id" = $1 ORDER BY "countries"."id" ASC LIMIT 1 [["approval_status_id", 2]]
However, the countries returned still list their invalid cities as the invalid cities aren't filtered from within the valid countries.
P.S. Let me know if the entire style is off here as I'm learning rails on my own.
I think you could check for the reverse. Check if there's any activity, city or country that's invalid. You just need to run a
find_by
. If it returnsnil
then you're golden, if it returns an object then it's invalid.Good RoRing