Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[request] schema-aware disabling of referential integrity while using fixtures #31

Open
mlt opened this issue Feb 8, 2019 · 15 comments

Comments

@mlt
Copy link
Contributor

mlt commented Feb 8, 2019

The problem is that when using fixtures for testing, rails disables referential integrity by using table names only. This requires putting all necessary schema names into a schema_search_path in config/database.yml. This is quite a PITA to remember. It would be nice if it was done automagically.

Rails calls ActiveRecord::ConnectionAdapters::PostgreSQL::ReferentialIntegrity#disable_referential_integrity
that uses ActiveRecord::ConnectionAdapters::SchemaStatements#tables which returns table names only without schema name to be quoted in #disable_referential_integrity. I feel like something could be done by returning already enquoted (as in PG format() with %I) name within ActiveRecord::ConnectionAdapters::PostgreSQL::SchemaStatements#data_source_sql or somewhere there.

I'm not sure how doable it is or whether it is better to file a bug somewhere upstream, but I'll leave it here as an idea.

@mlt
Copy link
Contributor Author

mlt commented Feb 8, 2019

Aparently, there is rails/rails#30846

@crashtech
Copy link
Owner

crashtech commented Feb 19, 2019

The fix seems reasonable, maybe I can push a new version with this improvement, just as one "good to have" since only in new Rails versions this might get fixed.

Does this work for you?

@mlt
Copy link
Contributor Author

mlt commented Feb 25, 2019

It does thus far! I would use the one from 5.2 branch not to require newer ruby https://github.com/mlt/rails/commits/bug-30846-5.2 . It also uses reduced filter for schema names.

@mlt
Copy link
Contributor Author

mlt commented Feb 26, 2019

Hold on! I need to include similar stuff to enable triggers back :-)

@mlt
Copy link
Contributor Author

mlt commented Mar 5, 2019

I think it is good now! Same link, same branch.

@crashtech
Copy link
Owner

Hey @mlt, I'm finally with time to check this out. Was this fixed on Rails or do you think it's still a valid feature?

@mlt
Copy link
Contributor Author

mlt commented Mar 28, 2019

It does not look like there is any action on this one upstream 😞 I personally still use it as it works for me. If you can make it into a monkey patch or something, that would be great! I just re-based both 5.2 and master branches.

@mlt
Copy link
Contributor Author

mlt commented Jul 17, 2019

Are you working on this or would you rather prefer for me to attempt to create a PR for both my requests?

@crashtech
Copy link
Owner

Right now I'm not being able to work on this. I think it's worth creating a PR. I have a suggestion that might work and won't break anything.

# on app/models/user.rb
module User
  extend Torque::PostgreSQL::SchemaAwareness
  # Or something like this, just to make sure we manually set the base modules that should be used as schema
end

# then while building the table name, we can check if the class is under a module other than Object, and if that is a Module which also is schema_aware?

@mlt
Copy link
Contributor Author

mlt commented Jul 18, 2019

I think what I have does not break anything… at least for me. I do not quite follow what is the point. The only downside of my approach that it unconditionally messes with triggers for all tables in all schemas. This might be heavy, I agree. But that is what Rails is doing for public only anyway.
I am not sure how other approach using schema. I personally have

module MyMod
  def self.table_name_prefix; 'my_schema.' end
...

So if we want to minimize number of tables we are messing with (for legacy DBs with tables not necessarily having Rails models) we can check whether self.table_name contains dot. Would not explicitly extending SchemaAwareness be an overkill? But once again this and/or your approach entail going over/enumerating all models. I feel like this is also heavy. I mean there might be a reason to just query pg_class and be done with it.

@crashtech
Copy link
Owner

The only problem with your approach is that it breaks some Rails rules, which we should try to follow to make sure that we definitely not gonna break anything. And I also think that SchemaAwareness it's way easier to control and change or add behaviors.

We need to use arel and table name quoting to guarantee the best compatibility. I know it can be an overkill, but there are rules and standards to follow. Besides, my idea with this GEM is that in the future, some of these features could be integrated on ActiveRecord with just a copy and paste (that's why it has this odd folder structure).

mlt added a commit to mlt/torque-postgresql that referenced this issue Jul 19, 2019
@mlt
Copy link
Contributor Author

mlt commented Jul 19, 2019

I think you overestimate my understanding of Rails 😇 I get the whole point of doing things "right" though.

I think I created a monster 😞 It works… but it is ugly as hell. We need to weed out views. The only way I see it either through some marker in model class or filter out by querying the DB. Another can of worms is models using different connections within a schema-aware module.

@mlt
Copy link
Contributor Author

mlt commented Aug 12, 2019

Also the other day I saw this https://stackoverflow.com/a/49584660/673826 but I didn't try. I wonder how neat it would be to use that.

@crashtech
Copy link
Owner

Rails 6 solves this problem, so I think there's no point to move forward with this feature. What do you think?

@mlt
Copy link
Contributor Author

mlt commented Aug 8, 2020

Nope, can't confirm. It is still an issue with Rails 6.0.3.2. I've set up a sample repo to demonstrate the problem mlt/deleteme@e69df1c

bundle exec rake db:setup RAILS_ENV=test
bundle exec rails db:migrate RAILS_ENV=test
bundle exec rails test
bundle exec rails test

Upon succeeding re-runs, you'd see

WARNING: Rails was not able to disable referential integrity.

This is most likely caused due to missing permissions.
Rails needs superuser privileges to disable referential integrity.

    cause: 

Run options: --seed 34209

# Running:

E

Error:
TTest#test_should_pass:
ActiveRecord::InvalidForeignKey: PG::ForeignKeyViolation: ERROR:  update or delete on table "ts" violates foreign key constraint "ts_t_id_fkey" on table "ts"
DETAIL:  Key (t_id)=(1) is still referenced from table "ts".

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants