116👍
You want to use the csv module that is part of the python language and you should use Django’s get_or_create method
with open(path) as f:
reader = csv.reader(f)
for row in reader:
_, created = Teacher.objects.get_or_create(
first_name=row[0],
last_name=row[1],
middle_name=row[2],
)
# creates a tuple of the new object or
# current object and a boolean of if it was created
In my example the model teacher has three attributes first_name, last_name and middle_name.
Django documentation of get_or_create method
44👍
If you want to use a library, a quick google search for csv
and django
reveals two libraries – django-csvimport and django-adaptors. Let’s read what they have to say about themselves…
- django-adaptors:
Django adaptor is a tool which allow you to transform easily a CSV/XML
file into a python object or a django model instance.
- django-importcsv:
django-csvimport is a generic importer tool to allow the upload of CSV
files for populating data.
The first requires you to write a model to match the csv file, while the second is more of a command-line importer, which is a huge difference in the way you work with them, and each is good for a different type of project.
So which one to use? That depends on which of those will be better suited for your project in the long run.
However, you can also avoid a library altogether, by writing your own django script to import your csv file, something along the lines of (warning, pseudo-code ahead):
# open file & create csvreader
import csv, yada yada yada
# import the relevant model
from myproject.models import Foo
#loop:
for line in csv file:
line = parse line to a list
# add some custom validation\parsing for some of the fields
foo = Foo(fieldname1=line[1], fieldname2=line[2] ... etc. )
try:
foo.save()
except:
# if the're a problem anywhere, you wanna know about it
print "there was a problem with line", i
It’s super easy. Hell, you can do it interactively through the django shell if it’s a one-time import. Just – figure out what you want to do with your project, how many files do you need to handle and then – if you decide to use a library, try figuring out which one better suits your needs.
- [Django]-How can I keep test data after Django tests complete?
- [Django]-UnicodeDecodeError: 'ascii' codec can't decode byte 0xd1 in position 2: ordinal not in range(128)
- [Django]-How can I get all the request headers in Django?
14👍
Use the Pandas library to create a dataframe of the csv data.
Name the fields either by including them in the csv file’s first line or in code by using the dataframe’s columns method.
Then create a list of model instances.
Finally use the django method .bulk_create() to send your list of model instances to the database table.
The read_csv function in pandas is great for reading csv files and gives you lots of parameters to skip lines, omit fields, etc.
import pandas as pd
from app.models import Product
tmp_data=pd.read_csv('file.csv',sep=';')
#ensure fields are named~ID,Product_ID,Name,Ratio,Description
#concatenate name and Product_id to make a new field a la Dr.Dee's answer
products = [
Product(
name = tmp_data.ix[row]['Name'],
description = tmp_data.ix[row]['Description'],
price = tmp_data.ix[row]['price'],
)
for row in tmp_data['ID']
]
Product.objects.bulk_create(products)
I was using the answer by mmrs151 but saving each row (instance) was very slow and any fields containing the delimiting character (even inside of quotes) were not handled by the open() — line.split(‘;’) method.
Pandas has so many useful caveats, it is worth getting to know
- [Django]-Running Django with FastCGI or with mod_python
- [Django]-Allowing RabbitMQ-Server Connections
- [Django]-How to use refresh token to obtain new access token on django-oauth-toolkit?
10👍
You can also use, django-adaptors
>>> from adaptor.model import CsvModel
>>> class MyCSvModel(CsvModel):
... name = CharField()
... age = IntegerField()
... length = FloatField()
...
... class Meta:
... delimiter = ";"
You declare a MyCsvModel which will match to a CSV file like this:
Anthony;27;1.75
To import the file or any iterable object, just do:
>>> my_csv_list = MyCsvModel.import_data(data = open("my_csv_file_name.csv"))
>>> first_line = my_csv_list[0]
>>> first_line.age
27
Without an explicit declaration, data and columns are matched in the same order:
Anthony --> Column 0 --> Field 0 --> name
27 --> Column 1 --> Field 1 --> age
1.75 --> Column 2 --> Field 2 --> length
- [Django]-How can I activate the unaccent extension on an already existing model
- [Django]-Can "list_display" in a Django ModelAdmin display attributes of ForeignKey fields?
- [Django]-Django-Admin: CharField as TextArea
10👍
For django 1.8 that im using,
I made a command that you can create objects dynamically in the future,
so you can just put the file path of the csv, the model name and the app name of the relevant django application, and it will populate the relevant model without specified the field names.
so if we take for example the next csv:
field1,field2,field3
value1,value2,value3
value11,value22,value33
it will create the objects
[{field1:value1,field2:value2,field3:value3}, {field1:value11,field2:value22,field3:value33}]
for the model name you will enter to the command.
the command code:
from django.core.management.base import BaseCommand
from django.db.models.loading import get_model
import csv
class Command(BaseCommand):
help = 'Creating model objects according the file path specified'
def add_arguments(self, parser):
parser.add_argument('--path', type=str, help="file path")
parser.add_argument('--model_name', type=str, help="model name")
parser.add_argument('--app_name', type=str, help="django app name that the model is connected to")
def handle(self, *args, **options):
file_path = options['path']
_model = get_model(options['app_name'], options['model_name'])
with open(file_path, 'rb') as csv_file:
reader = csv.reader(csv_file, delimiter=',', quotechar='|')
header = reader.next()
for row in reader:
_object_dict = {key: value for key, value in zip(header, row)}
_model.objects.create(**_object_dict)
note that maybe in later versions
from django.db.models.loading import get_model
is deprecated and need to be change to
from django.apps.apps import get_model
- [Django]-How to update fields in a model without creating a new record in django?
- [Django]-Where to put business logic in django
- [Django]-Django Server Error: port is already in use
8👍
The Python csv library can do your parsing and your code can translate them into Products()
.
- [Django]-Django vs. Model View Controller
- [Django]-Django: How to manage development and production settings?
- [Django]-Can I make an admin field not required in Django without creating a form?
7👍
something like this:
f = open('data.txt', 'r')
for line in f:
line = line.split(';')
product = Product()
product.name = line[2] + '(' + line[1] + ')'
product.description = line[4]
product.price = '' #data is missing from file
product.save()
f.close()
- [Django]-How can I filter a Django query with a list of values?
- [Django]-Django rest framework nested self-referential objects
- [Django]-Django: accessing session variables from within a template?
7👍
Write command in Django app. Where you need to provide a CSV file and loop it and create a model with every new row.
your_app_folder/management/commands/ProcessCsv.py
import os
from django.core.management.base import BaseCommand
from django.conf import settings
from your_app_name.models import Product
class Command(BaseCommand):
def handle(self, *args, **options):
with open(os.path.join(settings.BASE_DIR / 'your_csv_file.csv'), 'r') as csv_file:
csv_reader = csv.reader(csv_file, delimiter=';')
for row in csv_reader:
Product.objects.create(name=row[2], description=row[3], price=row[4])
At the end just run the command to process your CSV file and insert it into Product
model.
Terminal:
python manage.py ProcessCsv
Thats it.
- [Django]-Disable migrations when running unit tests in Django 1.7
- [Django]-How to get form fields' id in Django
- [Django]-What is the path that Django uses for locating and loading templates?
5👍
If you’re working with new versions of Django (>10) and don’t want to spend time writing the model definition. you can use the ogrinspect tool.
This will create a code definition for the model .
python manage.py ogrinspect [/path/to/thecsv] Product
The output will be the class (model) definition. In this case the model will be called Product.
You need to copy this code into your models.py file.
Afterwards you need to migrate (in the shell) the new Product table with:
python manage.py makemigrations
python manage.py migrate
More information here:
https://docs.djangoproject.com/en/1.11/ref/contrib/gis/tutorial/
Do note that the example has been done for ESRI Shapefiles but it works pretty good with standard CSV files as well.
For ingesting your data (in CSV format) you can use pandas.
import pandas as pd
your_dataframe = pd.read_csv(path_to_csv)
# Make a row iterator (this will go row by row)
iter_data = your_dataframe.iterrows()
Now, every row needs to be transformed into a dictionary and use this dict for instantiating your model (in this case, Product())
# python 2.x
map(lambda (i,data) : Product.objects.create(**dict(data)),iter_data
Done, check your database now.
- [Django]-How can I find the union of two Django querysets?
- [Django]-Django create userprofile if does not exist
- [Django]-Unable to find a locale path to store translations for file __init__.py
4👍
You can use the django-csv-importer package.
http://pypi.python.org/pypi/django-csv-importer/0.1.1
It works like a django model
MyCsvModel(CsvModel):
field1 = IntegerField()
field2 = CharField()
etc
class Meta:
delimiter = ";"
dbModel = Product
And you just have to:
CsvModel.import_from_file(“my file”)
That will automatically create your products.
- [Django]-How do I clone a Django model instance object and save it to the database?
- [Django]-Stack trace from manage.py runserver not appearing
- [Django]-In a django model custom save() method, how should you identify a new object?
4👍
You can give a try to django-import-export. It has nice admin integration, changes preview, can create, update, delete objects.
- [Django]-POST jQuery array to Django
- [Django]-Is it better to use path() or url() in urls.py for django 2.0?
- [Django]-Django: How to get related objects of a queryset?
3👍
This is based off of Erik’s answer from earlier, but I’ve found it easiest to read in the .csv file using pandas and then create a new instance of the class for every row in the in data frame.
This example is updated using iloc
as pandas
no longer uses ix in the most recent version. I don’t know about Erik’s situation but you need to create the list outside of the for loop otherwise it will not append to your array but simply overwrite it.
import pandas as pd
df = pd.read_csv('path_to_file', sep='delimiter')
products = []
for i in range(len(df)):
products.append(
Product(
name=df.iloc[i][0]
description=df.iloc[i][1]
price=df.iloc[i][2]
)
)
Product.objects.bulk_create(products)
This is just breaking the DataFrame into an array of rows and then selecting each column out of that array off the zero index. (i.e. name is the first column, description the second, etc.)
Hope that helps.
- [Django]-How to put comments in Django templates?
- [Django]-How to specify an IP address with Django test client?
- [Django]-Reload django object from database
- [Django]-How to change a django QueryDict to Python Dict?
- [Django]-Django admin ManyToMany inline "has no ForeignKey to" error
- [Django]-Python Django Gmail SMTP setup
1👍
Consider using Django’s built-in deserializers. Django’s docs are well-written and can help you get started. Consider converting your data from csv to XML or JSON and using a deserializer to import the data. If you’re doing this from the command line (rather than through a web request), the loaddata
manage.py command will be especially helpful.
- [Django]-Django rest framework change primary key to use a unqiue field
- [Django]-How do I migrate a model out of one django app and into a new one?
- [Django]-Django REST Framework: how to substitute null with empty string?
1👍
define class in models.py and a function in it.
class all_products(models.Model):
def get_all_products():
items = []
with open('EXACT FILE PATH OF YOUR CSV FILE','r') as fp:
# You can also put the relative path of csv file
# with respect to the manage.py file
reader1 = csv.reader(fp, delimiter=';')
for value in reader1:
items.append(value)
return items
You can access ith element in the list as items[i]
- [Django]-How do I get user IP address in Django?
- [Django]-How to iterate through dictionary in a dictionary in django template?
- [Django]-Factory-boy create a list of SubFactory for a Factory
0👍
If you’re using Postgres
& psycopg2
, you can use the Cursor
object to import text files directly –
from io import BytesIO
from django.db import connection
table = "<Table Name>"
columns = ["<Column Names>"]
file = BytesIO(
b"""
1;"02-01-101101";"Worm Gear HRF 50";"Ratio 1 : 10";"input shaft, output shaft, direction A, color dark green";
2;"02-01-101102";"Worm Gear HRF 50";"Ratio 1 : 20";"input shaft, output shaft, direction A, color dark green";
3;"02-01-101103";"Worm Gear HRF 50";"Ratio 1 : 30";"input shaft, output shaft, direction A, color dark green";
4;"02-01-101104";"Worm Gear HRF 50";"Ratio 1 : 40";"input shaft, output shaft, direction A, color dark green";
5;"02-01-101105";"Worm Gear HRF 50";"Ratio 1 : 50";"input shaft, output shaft, direction A, color dark green";
"""
)
with connection.cursor() as cursor:
cursor.copy_from(
file=file,
table=table,
sep=';',
columns=columns,
)
Inspiration: https://hakibenita.com/fast-load-data-python-postgresql
- [Django]-Can I make an admin field not required in Django without creating a form?
- [Django]-How to get the domain name of my site within a Django template?
- [Django]-Homepage login form Django