Как я могу создать сокращитель URL-адресов с помощью Django на основе идентификатора URL?

#python #django #sqlite #url-shortener

Вопрос:

Я работаю с Django, и я всего лишь новичок. Теперь я пытаюсь сделать сокращение URL-адреса с базой 62, поэтому я создал один класс, models.py и его имя таково URLGenerator . Вот его код:

 class URLGenerator:
   BASE = 62
   UPPERCASE_OFFSET = 55
   LOWERCASE_OFFSET = 61
   DIGIT_OFFSET = 48

   def generate_unique_key(self, integer):
      """
      Turn an integer [integer] into a base [BASE] number
       in string representation
      """

      # we won't step into the while if integer is 0
      # so we just solve for that case here
      if integer == 0:
         return '0'

     string = ""
     remainder: int = 0
     while integer > 0:
        remainder = integer % self.BASE
        string = self._true_chr(remainder)   string
        integer = int(integer / self.BASE)
    return string

   def get_id(self, key):
      """
      Turn the base [BASE] number [key] into an integer
      """
      int_sum = 0
      reversed_key = key[::-1]
      for idx, char in enumerate(reversed_key):
        int_sum  = self._true_ord(char) * int(math.pow(self.BASE, idx))
    return int_sum

   def _true_ord(self, char):
      """
      Turns a digit [char] in character representation
      from the number system with base [BASE] into an integer.
      """

     if char.isdigit():
          return ord(char) - self.DIGIT_OFFSET
     elif 'A' <= char <= 'Z':
         return ord(char) - self.UPPERCASE_OFFSET
     elif 'a' <= char <= 'z':
        return ord(char) - self.LOWERCASE_OFFSET
     else:
         raise ValueError("%s is not a valid character" % char)

   def _true_chr(self, integer):
      """
      Turns an integer [integer] into digit in base [BASE]
      as a character representation.
      """
     if integer < 10:
         return chr(integer   self.DIGIT_OFFSET)
     elif 10 <= integer <= 35:
         return chr(integer   self.UPPERCASE_OFFSET)
     elif 36 <= integer < 62:
         return chr(integer   self.LOWERCASE_OFFSET)
     else:
         raise ValueError(
            "%d is not a valid integer in the range of base %d" % (integer, BASE))
 

В приведенном выше классе есть два важных метода. Первый-это тот generate_unique_key , который может преобразовать целое число в уникальную строку, а второй-тот get_id , который может преобразовать строку в int. Например, что-то в этом роде:

 id = 1024
generator = URLGenerator()
key = generator.generate_unique_key(id) # key = GW
idx = generator.get_id(key)             # idx = 1024

# so the final url should look like this:
# http://localhost:8000/GW
 

В приведенном выше примере на самом деле id это id запись в базе данных, которую я хочу использовать во время сохранения URL-адреса в базе данных. Затем, если пользователь пишет http://127.0.0.1:8000/GW он должен быть перенаправлен на основной URL-адрес, который http://127.0.0.1:8000/1024.

И у меня есть еще один класс, в котором есть только одно поле :

 class Url(models.Model):
   url_id = models.AutoField(primary_key=True)

   def save(self , *args , **kwargs):
      pass
 

Но теперь я не знаю, как вызвать методы URLGenerator в моем URL классе, а затем написать простую функцию, используя POST in views.py , чтобы увидеть результат.

Я прочитал некоторые сообщения в StackOverflow, а также прочитал несколько руководств, но, к сожалению, я не смог решить эту проблему. Я буду благодарен вам за помощь и советы.

Ответ №1:

Во-первых, лучше добавить в Url модель еще одно поле, которое было бы примерно таким:

 url_id = models.AutoField(primary_key=True)
 

И почему мы должны его спасать? потому что в этом вопросе мы собираемся создать короткую базу URL-адресов на основе идентификатора каждой записи в нашей базе данных. Затем я создаю простую форму в forms.py . Вот его код:

 from django.forms import ModelForm, fields
from .models import Url


class URLForm(ModelForm):
    class Meta:
        model = Url
        fields = ['link']
        labels = {
            'link' : 'Enter link'
        }
 

Наконец, у меня есть две функции в моей views.py . В первом из них я использовал POST метод и написал простой запрос для сохранения данных в базе данных:

 # Redirect short URL to its original URL, if it's valid
def redirector_view(request):
    form = URLForm()
    if request.method == "POST":
        form = URLForm(request.POST)
        if form.is_valid():
            link = form.cleaned_data['link']
            if("http://" not in link) and ("https://" not in link):
                link = "http://"   link
            # uid = 
            new_url_obj = Url(link=link)
            new_url_obj.save()
            id = new_url_obj.pk
            url_generator_obj = URLGenerator()
            sh_url = url_generator_obj.generate_unique_key(id)
            new_url = f"http://127.0.0.1:8000/url/final/{sh_url}"
            json_object = {
                "final_url" : new_url
            }
            return JsonResponse(json_object)
    context = {
        "form" : form
    }
    return HttpResponse(context)
 

И вторая функция делает именно то, что я сказал в своем вопросе:

 # id = 1024
# generator = URLGenerator()
# key = generator.generate_unique_key(id) # key = GW
# idx = generator.get_id(key)   

def final(request , sh_url):
    url_generator_obj = URLGenerator()
    id = url_generator_obj.get_id(sh_url)
    url_details = Url.objects.get(url_id=id)
    return redirect(url_details.link)
 

И не забудьте добавить urls.py файл в приложение :

 from django.urls import path
from . import views

urlpatterns = [
    path('', views.redirector_view, name="urlshortner"),
    path('final/<str:sh_url>' , views.final , name="final"),
]
 

Теперь, если я добавлю в свою базу данных простую запись, которая содержит только URL-адрес, ее pk будет 1(потому что это наш первый объект), а затем добавьте 1 к этому URL-адресу:
http://127.0.0.1:8000/url/final/
вы будете перенаправлены на основной URL-адрес.