Загрузите удаленное изображение и сохраните его в модели Django - программирование
Подтвердить что ты не робот

Загрузите удаленное изображение и сохраните его в модели Django

Я пишу приложение Django, которое будет извлекать все изображения определенного URL-адреса и сохранять их в базе данных.

Но я не понимаю, как использовать ImageField в Django.

Settings.py

MEDIA_ROOT = os.path.join(PWD, "../downloads/")

# URL that handles the media served from MEDIA_ROOT. Make sure to use a
# trailing slash.
# Examples: "http://example.com/media/", "htp://media.example.com/"
MEDIA_URL = '/downloads/'

models.py

class images_data(models.Model):
        image_id =models.IntegerField()
        source_id = models.IntegerField()
        image=models.ImageField(upload_to='images',null=True, blank=True)
        text_ind=models.NullBooleanField()
        prob=models.FloatField()

download_img.py

def spider(site):
        PWD = os.path.dirname(os.path.realpath(__file__ ))
        #site="http://en.wikipedia.org/wiki/Pune"
        hdr= {'User-Agent': 'Mozilla/5.0'}
        outfolder=os.path.join(PWD, "../downloads")
        #outfolder="/home/mayank/Desktop/dreamport/downloads"
        print "MAYANK:"+outfolder
        req = urllib2.Request(site,headers=hdr)
        page = urllib2.urlopen(req)
        soup =bs(page)
        tag_image=soup.findAll("img")
        count=1;
        for image in tag_image:
                print "Image: %(src)s" % image
                filename = image["src"].split("/")[-1]
                outpath = os.path.join(outfolder, filename)
                urlretrieve('http:'+image["src"], outpath)
                im = img(image_id=count,source_id=1,image=outpath,text_ind=None,prob=0)
                im.save()
                count=count+1

Я вызываю download_imgs.py внутри одного вида, например

        if form.is_valid():
                url = form.cleaned_data['url']
                spider(url)
4b9b3361

Ответ 1

Документация Django - это всегда хорошее место для запуска

class ModelWithImage(models.Model):
    image = models.ImageField(
        upload_to='images',
    )

ОБНОВЛЕНО

Так работает script.

  • Прокрутка изображений для загрузки
  • Скачать изображение
  • Сохранить в файл temp
  • Применить к модели
  • Сохранить модель

.

import requests
import tempfile

from django.core import files

# List of images to download
image_urls = [
    'http://i.thegrindstone.com/wp-content/uploads/2013/01/how-to-get-awesome-back.jpg',
]

for image_url in image_urls:
    # Steam the image from the url
    request = requests.get(image_url, stream=True)

    # Was the request OK?
    if request.status_code != requests.codes.ok:
        # Nope, error handling, skip file etc etc etc
        continue

    # Get the filename from the url, used for saving later
    file_name = image_url.split('/')[-1]

    # Create a temporary file
    lf = tempfile.NamedTemporaryFile()

    # Read the streamed image in sections
    for block in request.iter_content(1024 * 8):

        # If no more file then stop
        if not block:
            break

        # Write image block to temporary file
        lf.write(block)

    # Create the model you want to save the image to
    image = Image()

    # Save the temporary image to the model#
    # This saves the model so be sure that is it valid
    image.image.save(file_name, files.File(lf))

Некоторые ссылочные ссылки:

  • requests - "HTTP для людей", я предпочитаю это urllib2
  • tempfile - сохранить файл временного файла, а не диск
  • Django filefield save

Ответ 2

Если вы хотите сохранить загруженные изображения, не сохраняя их сначала на диске (без использования NamedTemporaryFile и т.д.), тогда есть простой способ сделать это.

Это будет немного быстрее, чем загрузка файла и запись его на диск, поскольку все это делается в памяти. Обратите внимание, что этот пример написан для Python 3 - процесс аналогичен в Python 2, но немного отличается.

from django.core import files
from io import BytesIO
import requests

url = "https://example.com/image.jpg"
resp = requests.get(url)
if resp.status_code != requests.codes.ok:
    #  Error handling here

fp = BytesIO()
fp.write(resp.content)
file_name = url.split("/")[-1]  # There probably a better way of doing this but this is just a quick example
your_model.image_field.save(file_name, files.File(fp))

Где your_model - это экземпляр модели, которую вы хотите сохранить, и .image_field - это имя ImageField.

Дополнительную информацию см. в документации для io.

Ответ 3

В качестве примера того, о чем я думаю, вы спрашиваете:

В forms.py:

imgfile = forms.ImageField(label = 'Choose your image', help_text = 'The image should be cool.')

В models.py:

imgfile =   models.ImageField(upload_to='images/%m/%d')

Таким образом, у пользователя будет запрос POST (когда пользователь заполнит форму). Этот запрос будет содержать в основном словарь данных. Словарь содержит предоставленные файлы. Чтобы сфокусировать запрос на файл с поля (в нашем случае, ImageField), вы должны использовать:

request.FILES['imgfield']

Это можно использовать при создании объекта модели (создание экземпляра класса модели):

newPic = ImageModel(imgfile = request.FILES['imgfile'])

Чтобы сохранить этот простой способ, вы просто используете метод save(), предоставленный вашему объекту (поскольку Django - это замечательно):

if form.is_valid():
    newPic = Pic(imgfile = request.FILES['imgfile'])
    newPic.save()

Ваше изображение будет сохранено по умолчанию в каталоге, который вы указываете для MEDIA_ROOT в settings.py.

Доступ к изображению в шаблоне:

<img src="{{ MEDIA_URL }}{{ image.imgfile.name }}"></img>

URL-адреса могут быть сложными, но вот базовый пример простого шаблона url для вызова сохраненных изображений:

urlpatterns += patterns('',
        url(r'^media/(?P<path>.*)$', 'django.views.static.serve', {
            'document_root': settings.MEDIA_ROOT,
        }),
   )

Надеюсь, это поможет.

Ответ 4

Попробуйте сделать это, вместо того, чтобы назначать путь к изображению...

    import urllib2
    from django.core.files.temp import NamedTemporaryFile
    def handle_upload_url_file(url):
        img_temp = NamedTemporaryFile()
        opener = urllib2.build_opener()
        opener.addheaders = [('User-agent', 'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:15.0) Gecko/20120427 Firefox/15.0a1')]
        img_temp.write(opener.open(url).read())
        img_temp.flush()
        return img_temp

используйте вышеуказанную функцию, как это.

    new_image = images_data()
    #rest of the data in new_image and then do this.
    new_image.image.save(slug_filename,File(handle_upload_url_file(url)))
    #here slug_filename is just filename that you want to save the file with.

Ответ 5

Как и в случае с @boltsfrombluesky, вы можете сделать это в Python 3 без каких-либо внешних зависимостей:

from os.path import basename
import urllib.request
from urllib.parse import urlparse
import tempfile

from django.core.files.base import File

def handle_upload_url_file(url, obj):
    img_temp = tempfile.NamedTemporaryFile(delete=True)
    req = urllib.request.Request(
        url, data=None,
        headers={
            'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_9_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/35.0.1916.47 Safari/537.36'
        }
    )
    with urllib.request.urlopen(req) as response:
        img_temp.write(response.read())
    img_temp.flush()
    filename = basename(urlparse(url).path)
    result = obj.image.save(filename, File(img_temp))
    img_temp.close()
    return result

Ответ 6

def qrcodesave(request): 
    import urllib2;   
    url ="http://chart.apis.google.com/chart?cht=qr&chs=300x300&chl=s&chld=H|0"; 
    opener = urllib2.urlopen(url);  
    mimetype = "application/octet-stream"
    response = HttpResponse(opener.read(), mimetype=mimetype)
    response["Content-Disposition"]= "attachment; filename=aktel.png"
    return response