MVC multi-model data import (fat model?)
I'm developing an app that will need to import data for three models from a single CSV file (with one-to-many associations). I have set up a Datafile model & controller to handle upload / parsing of the file. Right now, all the logic for parsing and saving the records is in the controller. This allows me to save to several different models, get the IDs for the saved records, and create associations as necessary while the file is being parsed.
Thinking about the "fat model, skinny controller" principle, I realized I have about 150 lines of code in a controller that is really just processing data. As I started looking at moving this to a model, though, I concluded I would have to process all of this data into arrays (without knowing association IDs), and send it back to the controller to be saved (since the model can't call methods from other models). I'm anticipating having about 1,500 records in the import file. I'm using CakePHP, which has a saveAll() method to save data to several models sim开发者_运维知识库ultaneously from a single array.
Another option would be to have each of the three models separately parse the file, ignoring any data it doesn't need. This should be possible, as long as I send it to the models in the right order, and give the "belongsTo" models a list of possibly associated records to search.
So -- any advice on these options?
- Leave the parsing code in the Datafile controller as it is.
- Move all the parsing code to the Datafile model, then pass back a large array to be saved through the Datafile controller.
- Send the file separately to each of the three models, along with supplemental lists for determining associations.
For import scripts I'd always prefer a shell script. A controller is probably the worst possible place to put complicated import logic, since it's almost impossible to reuse.
If you don't have a problem with running the script from the CLI, use a Shell. You can even invoke it from a web interface, if need be.
Otherwise, put the logic into the model, to be able to call it from both the web and the CLI.
I built something similar to this, however it was not model-specific. It allows you to upload and download csv sheets that use multiple models in the same sheet. I used a series of components that called the models I wanted to upload as CSV sheets.
I have a rawdata component that processes files line by line. On each line it does a callback to the controller for before_process_row and after_process_row, etc. The controller callback then calls another component to do the splicing up of the data into the different models based on rules.
The only information I have in the models are rules for what fields display in the csv file and the file header information and other search criteria.
It may be a little more complex than you're looking for, but I went with components for the data processing part.
Is this an operation that will only be performed once? If so, write a script to do it. Otherwise....
I'd import all of the data into the primary controller, construct the data array (in the same format as $this->data, then save it either with saveAll or individual save depending on which is more appropriate. So long as it's saved with the correct relations, it doesn't really matter how you do it!
Remember to do a $this->myModel->create() before each one to ensure that you don't just end up with one row.
 
         加载中,请稍侯......
 加载中,请稍侯......
      
精彩评论