How do I get ActiveRecord to show the next id (last + 1) in Ruby on Rails?

前端 未结 13 1233
臣服心动
臣服心动 2020-12-05 06:33

Is there a compact way with ActiveRecord to query for what id it\'s going to use next if an object was going to be persisted to the database? In SQL, a query like this would

相关标签:
13条回答
  • 2020-12-05 07:03

    I don't think there's a generic answer to this since you may not want to assume one database over another. Oracle, for example, may assign id's by a sequence or, worse, by a trigger.

    As far as other databases are concerned, one may set them up with non sequential or random id allocation.

    May I ask what the use case is? Why would you need to anticipate the next id? What does 'next' mean? Relative to when? What about race conditions (multiuser environments)?

    0 讨论(0)
  • 2020-12-05 07:03

    If you want to be sure no one else can take the 'new' index, you should lock the table. By using something like:

    ActiveRecord::Base.connection.execute("LOCK TABLES table_name WRITE")
    

    and

    ActiveRecord::Base.connection.execute("UNLOCK TABLES")
    

    But this is specific for each database engine.

    The only correct answer for a sequential id column is:

    YourModel.maximum(:id)+1
    

    If you sort your model in a default scope, last and first will depend on that order.

    0 讨论(0)
  • 2020-12-05 07:04

    I came to this SO question because I wanted to be able to predict the id of a model created in my test suite (the id was then used a REST request to an external service and I needed to predict the exact value to mock the request).

    I found that Model.maximum(:id).next, although elegant, doesn't work in a rails testing environment with transactional fixtures since there are usually no records in the db so it will simply return nil.

    Transactional fixtures make the issue extra tricky since the auto increment field ascends even when there aren't any records in the db. Furthermore using an ALTER TABLE ***your_table_name*** AUTO_INCREMENT = 100 breaks the transaction your tests are in because it requires its own transaction.

    What I did to solve this was to create a new object and add 1 to its id:

    let!(:my_model_next_id) { FactoryBot.create(:my_model).id + 1 }
    

    Although somewhat hacky (and slightly inefficient on your db since you create an extra object for the sake of its id), it doesn't do anything goofy to the transaction and works reliably in a testing environment with no records (unless your tests run in parallel with access to the same db...in which case: race conditions...maybe?).

    0 讨论(0)
  • 2020-12-05 07:13

    Don't get the intent but you might wanna think of using GUID

    0 讨论(0)
  • 2020-12-05 07:14

    You should never assume what the next id will be in the sequence. If you have more than 1 user you run the risk of the id being in use by the time the new object is created.

    Instead, the safe approach would be to create the new object and update it. Making 2 hits to your database but with an absolute id for the object you're working with.

    This method takes the guess work out of the equation.

    0 讨论(0)
  • 2020-12-05 07:15

    If your database is Postgres, you can get the next id with this (example for a table called 'users'):

    ActiveRecord::Base.connection.execute("select last_value from users_id_seq").first["last_value"]
    

    Unlike the other answers, this value is not affected by the deletion of records.

    There's probably a mySQL equivalent, but I don't have one set up to confirm.

    If you have imported data into your postgresql database, there's a good chance that the next id value after the import is not set to the next integer greater than the largest one you imported. So you will run into problems trying to save the activerecord model instances.

    In this scenario, you will need to set the next id value manually like this:

    ActiveRecord::Base.connection.execute("alter sequence users_id_seq restart with 54321;") #or whatever value you need
    
    0 讨论(0)
提交回复
热议问题