PlayBox is a Django e-commerce site for gaming gear and accessories. It offers a user-friendly interface for browsing and purchasing gaming products, with a focus on delivering an engaging shopping experience.
- Backend: Django
- Styling: TailwindCSS
- Hosting: Pacil Web Service (PWS)
Explore the live version here: PlayBox
Langkah pertama yang saya lakukan adalah membuat proyek Django baru. Saya menggunakan perintah django-admin startproject
untuk menginisialisasi proyek dengan nama playbox
. Perintah ini menciptakan struktur dasar proyek Django yang meliputi beberapa file konfigurasi dan folder.
django-admin startproject playbox .
Setelah proyek berhasil dibuat, langkah berikutnya adalah menambahkan aplikasi baru dengan nama main
. Dalam Django, aplikasi adalah bagian modular dari proyek yang memungkinkan pengembangan fitur secara terpisah. Saya menjalankan perintah python manage.py startapp main
untuk membuat aplikasi main
.
python manage.py startapp main
Selanjutnya, saya harus mengonfigurasi routing proyek untuk menyertakan aplikasi main
. Dalam file urls.py
proyek, saya menambahkan routing untuk aplikasi main
dengan menggunakan fungsi include
agar aplikasi dapat mengelola URL-nya sendiri. Kode yang saya tambahkan adalah sebagai berikut:
from django.urls import path, include
from django.contrib import admin
urlpatterns = [
path('admin/', admin.site.urls),
path('', include('main.urls')),
]
Untuk menyimpan data produk, saya mendefinisikan sebuah model Product
dalam file models.py
aplikasi main
. Model ini mencakup atribut-atribut seperti name
, price
, description
, dan stock
. Saya juga menambahkan created_at
dan updated_at
untuk melacak waktu pembuatan dan pembaruan. Berikut adalah kode model yang saya buat:
from django.db import models
class Product(models.Model):
name = models.CharField(max_length=255)
price = models.IntegerField()
description = models.TextField()
stock = models.PositiveIntegerField()
image = models.CharField(max_length=255)
created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)
def __str__(self):
return self.name
Saya kemudian membuat sebuah view dalam file views.py
yang bertugas untuk merender template HTML. Fungsi landing
ini akan menampilkan template yang menunjukkan nama aplikasi serta nama dan kelas. Berikut adalah kode untuk fungsi view tersebut:
from django.shortcuts import render
def landing(request):
return render(request, "landing/index.html")
Untuk memastikan bahwa view landing
dapat diakses melalui URL tertentu, saya menambahkan routing di urls.py
aplikasi main
. Routing ini mengarahkan URL root ('') ke fungsi landing
. Kode yang saya tambahkan adalah sebagai berikut:
from django.urls import path
from main.views import landing
urlpatterns = [
path('', landing, name='landing'),
]
Terakhir, saya melakukan deployment aplikasi ke Platform Web Services (PWS). Prosesnya meliputi beberapa langkah:
Saya memulai dengan mengklik "Create New Project" pada dashboard PWS dan mengisi kolom Project Name
dengan nama playbox
. Saya kemudian mengklik "Create New Project" untuk menyelesaikan proses pembuatan proyek.
Dalam file settings.py
, saya menambahkan URL deployment PWS ke dalam daftar ALLOWED_HOSTS
untuk mengizinkan akses dari domain tersebut:
...
ALLOWED_HOSTS = ["localhost", "127.0.0.1", "muhammad-fazil31-playbox.pbp.cs.ui.ac.id"]
...
Saya menjalankan perintah yang tercantum pada bagian "Project Command" di halaman PWS. Saat melakukan push ke PWS, saya memasukkan username dan password yang diberikan oleh PWS.
Di dashboard PWS, saya memantau status proyek yang saya buat. Jika statusnya "Building", berarti proyek masih dalam proses deployment. Jika statusnya "Running", berarti proyek telah berhasil dideploy dan dapat diakses melalui URL yang diberikan. Untuk melihat hasilnya, saya menekan tombol "View Project".
Berikut adalah bagan sederhana yang menggambarkan alur request client ke web aplikasi berbasis Django beserta responnya serta hubungan antara urls.py
, views.py
, models.py
, dan berkas HTML:
Client Request (e.g., GET /)
|
v
urls.py
|
v
views.py
|
v
models.py (jika diperlukan untuk data)
|
v
HTML Template
|
v
Client Response (HTML)
-
Client Request: Client (misalnya, browser) mengirimkan request ke server, seperti permintaan untuk halaman utama ('/').
-
urls.py: Django menerima request dan mencocokkan URL dengan routing yang didefinisikan di
urls.py
. Jika URL cocok, Django akan meneruskan request ke view yang sesuai. -
views.py: View yang sesuai (misalnya,
landing
) akan dipanggil untuk menangani request. View ini bisa melakukan logika tambahan atau berinteraksi dengan model. -
models.py: Jika view memerlukan data dari database, view akan menggunakan model yang didefinisikan di
models.py
untuk berinteraksi dengan database. -
HTML Template: View merender template HTML dengan data yang diambil. Template ini adalah file HTML yang berisi struktur halaman web.
-
Client Response: Hasil render dari template HTML dikirim kembali sebagai respons ke client.
Git adalah sistem kontrol versi yang sangat penting dalam pengembangan perangkat lunak. Git memungkinkan pengembang untuk melacak perubahan pada kode sumber seiring berjalannya waktu. Beberapa fungsi utama Git dalam pengembangan perangkat lunak adalah:
- Version Control: Git memungkinkan pengembang untuk melacak versi berbeda dari kode sumber, memudahkan untuk kembali ke versi sebelumnya jika diperlukan.
- Branching and Merging: Pengembang dapat membuat cabang (branch) untuk fitur baru atau perbaikan tanpa mempengaruhi kode utama. Setelah selesai, cabang dapat digabungkan (merge) kembali ke branch utama.
- Collaboration: Git mendukung kolaborasi antara beberapa pengembang dengan memungkinkan mereka untuk bekerja pada repositori yang sama secara bersamaan tanpa konflik.
- History Tracking: Git menyimpan riwayat perubahan, termasuk siapa yang membuat perubahan dan kapan perubahan tersebut dibuat, membantu dalam audit dan debugging.
Django sering dipilih sebagai framework pemula dalam pembelajaran pengembangan perangkat lunak karena beberapa alasan:
- Struktur dan Konvensi: Django menyediakan struktur yang jelas dan mengikuti prinsip "konvensi di atas konfigurasi", yang membantu pemula memahami arsitektur aplikasi web.
- Dokumentasi dan Komunitas: Django memiliki dokumentasi yang sangat baik dan komunitas yang besar, sehingga memudahkan pemula untuk menemukan bantuan dan solusi.
- Fitur Bawaan: Django dilengkapi dengan banyak fitur bawaan seperti autentikasi pengguna, panel admin, dan ORM, yang mempercepat pengembangan aplikasi tanpa harus membangun fitur-fitur dasar dari nol.
- Keamanan: Django secara default mengikuti praktik keamanan yang baik, mengurangi risiko kerentanan yang umum dalam pengembangan aplikasi web.
Model dalam Django disebut sebagai ORM (Object-Relational Mapping) karena Django menggunakan pendekatan ORM untuk memetakan objek Python ke tabel-tabel dalam database relasional. Dengan ORM, saya dapat menggunakan kode Python untuk berinteraksi dengan database tanpa menulis SQL secara langsung. ORM memudahkan operasi database seperti penyimpanan, pengambilan, pembaruan, dan penghapusan data melalui objek Python, sehingga meningkatkan produktivitas dan meminimalkan kemungkinan kesalahan.
Data delivery adalah proses menyampaikan data dari server ke client. Ini penting karena:
- Interaksi Dinamis: Aplikasi sering membutuhkan data terbaru yang dapat dikirim secara real-time atau sesuai permintaan dari client.
- Efisiensi dan Responsivitas: Data delivery memastikan bahwa client mendapatkan data yang relevan dan terkini dengan cara yang efisien.
- Pengalaman Pengguna: Pengalaman pengguna menjadi lebih baik ketika data dapat disampaikan dengan cepat dan akurat.
XML dan JSON adalah format untuk pertukaran data. JSON lebih populer dibandingkan XML karena:
- Sintaks yang lebih sederhana: JSON lebih ringkas dan mudah dibaca daripada XML.
- Ukuran yang lebih kecil: JSON biasanya menghasilkan file yang lebih kecil karena tidak memerlukan tag penutup seperti XML.
- Penggunaan yang luas dalam web dan API: JSON lebih mudah diintegrasikan dengan JavaScript dan banyak API modern menggunakan JSON karena kemudahan penggunaannya.
is_valid()
adalah metode pada form Django yang digunakan untuk memeriksa apakah data yang dikirimkan memenuhi semua validasi yang ditentukan dalam form.
- Fungsi: Mengembalikan
True
jika data form valid danFalse
jika tidak. Ini memeriksa validitas data berdasarkan aturan yang didefinisikan dalam form. - Kebutuhan: Memastikan bahwa data yang diproses aman dan sesuai dengan harapan sebelum disimpan atau diproses lebih lanjut.
CSRF Token adalah mekanisme untuk melindungi aplikasi web dari serangan Cross-Site Request Forgery (CSRF).
- Kebutuhan: Token CSRF diperlukan untuk memastikan bahwa permintaan yang dikirimkan ke server berasal dari pengguna yang sah.
- Tanpa Token: Jika tidak ada token CSRF, penyerang dapat membuat permintaan berbahaya menggunakan kredensial pengguna yang sah.
- Potensi Risiko: Penyerang dapat memanfaatkan kurangnya perlindungan CSRF untuk melakukan tindakan tidak sah seperti mengubah data pengguna atau mengirimkan permintaan yang merugikan.
Pertama, saya membuat berkas forms.py
dengan kode yang mendefinisikan struktur form untuk Product, lalu menambahkan kode pada views.py
untuk menangani form tersebut, serta memperbarui fungsi show_main
untuk menampilkan seluruh Product. Saya juga mengupdate urls.py
untuk menambahkan path ke fungsi create
, membuat berkas HTML create/index.html
untuk form input, dan menyesuaikan main.html
agar menampilkan data product dalam tabel serta menambahkan tombol untuk mengakses form input.
2. Tambahkan 4 fungsi views baru untuk melihat objek yang sudah ditambahkan dalam format XML, JSON, XML by ID, dan JSON by ID
Saya mengembalikan data dalam bentuk XML dan JSON dengan beberapa langkah. Pertama, saya membuka views.py
dan menambahkan import HttpResponse
dan serializers
. Saya membuat fungsi show_xml
untuk mengembalikan seluruh data MoodEntry dalam format XML dan fungsi show_json
untuk mengembalikan data yang sama dalam format JSON. Setelah itu, saya memperbarui urls.py
untuk menambahkan path URL yang sesuai. Selanjutnya, saya membuat fungsi show_xml_by_id
dan show_json_by_id
untuk mengembalikan data MoodEntry berdasarkan ID dalam format XML dan JSON, kemudian menambahkan path URL untuk kedua fungsi ini di urls.py
.
Saya menambahkan path URL ke dalam urlpatterns di berkas urls.py untuk menghubungkan URL yang sesuai dengan fungsi view yang telah dibuat. Saya menambahkan path "xml/"
untuk format XML dan "json/"
untuk format JSON. Untuk data berdasarkan ID, saya menambahkan path "xml/<str:id>/"
dan "json/<str:id>/"
.
Berikut adalah pengaturan urlpatterns yang saya buat:
urlpatterns = [
...
path("xml/", show_xml, name="show_xml"),
path("json/", show_json, name="show_json"),
path("xml/<str:id>/", show_xml_by_id, name="show_xml_by_id"),
path("json/<str:id>/", show_json_by_id, name="show_json_by_id"),
...
]
HttpResponseRedirect()
adalah kelas untuk membuat respon HTTP yang mengarahkan ke URL lain, sedangkan redirect()
adalah shortcut yang menyederhanakan penggunaan HttpResponseRedirect()
dan dapat menerima URL atau model instance.
Model Product memiliki field ForeignKey yang menunjuk ke model User, sehingga setiap produk terkait dengan pengguna tertentu. Ini memungkinkan kita untuk mengaitkan setiap produk dengan pemiliknya.
Authentication adalah proses verifikasi identitas pengguna (login), sedangkan authorization menentukan hak akses pengguna tersebut. Saat pengguna login, sistem melakukan authentication dengan memeriksa kredensialnya. Django mengimplementasikan ini melalui sistem autentikasi built-in.
Django menggunakan session untuk menyimpan informasi tentang pengguna yang login, dan secara otomatis mengatur cookie untuk menjaga session tersebut. Kegunaan lain dari cookies termasuk menyimpan preferensi pengguna dan informasi terakhir login. Tidak semua cookies aman; cookie harus disimpan dengan aman dan tidak boleh mengandung informasi sensitif.
Saya mulai dengan mengaktifkan virtual environment, kemudian menambahkan import di views.py
untuk UserCreationForm
, AuthenticationForm
, dan fungsi login/logout. Selanjutnya, saya membuat fungsi register
untuk menangani pendaftaran pengguna, membuat berkas register.html
, dan memperbarui urls.py
. Setelah itu, saya menambahkan fungsi login_user
dan membuat login.html
serta memperbarui urls.py
untuk login. Terakhir, saya menambahkan fungsi logout_user
, membuat tombol logout di main.html
, dan memperbarui urls.py
untuk logout.
Untuk membuat akun, saya hanya perlu melakukan registrasi dua kali dengan informasi pengguna yang berbeda. Setelah itu, untuk membuat dummy data, saya login dengan akun yang diinginkan. Setelah berhasil masuk, saya dapat menambahkan dummy data baru melalui form yang sudah dibuat pada Tugas 3 sebelumnya.
Saya menambahkan atribut user pada model Product, yang merupakan ForeignKey yang merujuk pada model User. Dengan cara ini, setiap produk akan terkait langsung dengan pengguna tertentu. Saya menggunakan parameter on_delete=models.CASCADE untuk memastikan bahwa jika seorang pengguna dihapus, semua produk yang mereka buat juga akan dihapus secara otomatis. Ini menjaga database saya tetap bersih dan teratur.
Berikut adalah implementasi lengkap dari model Product
:
class Product(models.Model):
...
user = models.ForeignKey(User, on_delete=models.CASCADE)
Pertama, saya mengedit views.py
dengan menambahkan import yang diperlukan. Dalam fungsi signin
, saya mengubah kode agar menambahkan cookie "last_login" dengan waktu saat pengguna login. Di fungsi landing
, saya menambahkan informasi cookie ke dalam konteks. Kode request.user.username
saya gunakan untuk menampilkan username pengguna di halaman utama. Saya juga menambahkan informasi cookie ke dalam context. Selain itu, saya mengubah fungsi logout
untuk menghapus cookie saat logout. Terakhir, saya memperbarui profile/index.html
untuk menampilkan sesi terakhir login.
Jika terdapat beberapa CSS selector untuk suatu elemen HTML, jelaskan urutan prioritas pengambilan CSS selector tersebut!
Prioritas pengambilan CSS selector mengikuti aturan sebagai berikut: inline styles memiliki prioritas tertinggi (misalnya style="color: blue;"
), diikuti oleh ID selector (#id
), yang lebih kuat dibandingkan class selector, attribute selector, dan pseudo-class. Di bawahnya, elemen (tag) memiliki prioritas terendah. Namun, penggunaan !important dapat mengabaikan prioritas selektor dan akan diutamakan.
Mengapa responsive design menjadi konsep yang penting dalam pengembangan aplikasi web? Berikan contoh aplikasi yang sudah dan belum menerapkan responsive design!
Responsive design penting karena memastikan bahwa aplikasi web dapat diakses dengan nyaman di berbagai perangkat (desktop, tablet, dan ponsel). Contoh aplikasi yang menerapkan responsive design adalah Google dan Twitter. Aplikasi yang tidak menerapkan responsive design mungkin memiliki tampilan yang tidak optimal pada perangkat tertentu, seperti aplikasi lama yang hanya didesain untuk desktop.
Jelaskan perbedaan antara margin, border, dan padding, serta cara untuk mengimplementasikan ketiga hal tersebut!
- Margin: Ruang di luar elemen, antara elemen dengan elemen lain.
- Border: Garis yang mengelilingi elemen, antara padding dan margin.
- Padding: Ruang di dalam elemen, antara konten dan border.
Ketiganya dapat diatur menggunakan CSS dengan sintaks seperti:
margin: 10px;
border: 1px solid black;
padding: 5px;
- Flexbox: Layout one-dimensional yang digunakan untuk mengatur elemen dalam satu baris atau kolom. Flexbox berguna untuk membuat layout yang fleksibel dan responsif dengan elemen-elemen yang dapat beradaptasi terhadap ukuran kontainer.
- Grid Layout: Layout two-dimensional yang memungkinkan kita untuk membuat struktur grid yang lebih kompleks, baik dalam baris maupun kolom, yang cocok untuk membuat desain yang lebih kompleks dan presisi tinggi.
Flexbox cocok untuk layout yang sederhana dan linier, sedangkan Grid lebih tepat untuk layout yang kompleks dan berbasis grid.
Untuk menambahkan fitur edit dan hapus produk dalam aplikasi, pertama-tama saya membuka file views.py
dan membuat dua fungsi edit
dan delete
seperti berikut:
@login_required(login_url="/signin")
def edit(request, id):
product = Product.objects.get(pk=id)
form = ProductForm(request.POST or None, request.FILES, instance=product)
if form.is_valid() and request.method == "POST":
form.save()
return HttpResponseRedirect(reverse("main:profile"))
context = {"form": form}
return render(request, "edit/index.html", context)
@login_required(login_url="/signin")
def delete(request, id):
product = Product.objects.get(pk=id)
product.delete()
return HttpResponseRedirect(reverse("main:profile"))
Fungsi edit
mengambil entri mood berdasarkan ID, menginisialisasi form, dan menyimpan perubahan jika valid sebelum mengarahkan ke halaman utama. Fungsi delete
mengambil dan menghapus entri mood berdasarkan ID, lalu kembali ke halaman utama. Saya menambahkan URL di urls.py
untuk kedua fungsi dengan path edit/<uuid:id>
dan delete/<uuid:id>
, serta tombol "Edit" dan "Delete" di template untuk setiap produk.
2. Kustomisasi desain pada template HTML yang telah dibuat pada tugas-tugas sebelumnya menggunakan CSS
Saya memulai proses kustomisasi desain pada aplikasi Django saya dengan menggunakan Tailwind CSS. Untuk menghubungkan template Django saya dengan Tailwind CSS, saya memanfaatkan script CDN dari Tailwind. Saya menambahkan script berikut di dalam tag <head>
di file base.html:
<head>
{% block meta %}
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1">
{% endblock meta %}
<script src="https://cdn.tailwindcss.com"></script>
</head>
Dengan menambahkan script ini, saya dapat menggunakan kelas-kelas Tailwind CSS di seluruh halaman web saya.
5. Untuk setiap card product, buatlah dua buah button untuk mengedit dan menghapus product pada card tersebut!
6. Buatlah navigation bar (navbar) untuk fitur-fitur pada aplikasi yang responsive terhadap perbedaan ukuran device, khususnya mobile dan desktop.
JavaScript adalah bahasa pemrograman yang sangat penting dalam pengembangan aplikasi web karena memungkinkan interaktivitas, pengolahan data asinkron, dan manipulasi DOM. Dengan JavaScript, pengembang dapat menciptakan elemen interaktif seperti tombol dan formulir, serta menggunakan AJAX untuk melakukan permintaan data tanpa memuat ulang halaman, sehingga meningkatkan pengalaman pengguna. Selain itu, JavaScript dapat mengubah struktur, gaya, dan konten halaman secara dinamis, memberikan pengalaman pengguna yang lebih kaya. Bahasa ini juga memiliki kompatibilitas lintas platform, dapat berjalan di semua browser modern, serta mendukung berbagai framework dan pustaka seperti React, Angular, dan Vue.js, yang membantu dalam pengembangan aplikasi web yang lebih kompleks dan terstruktur.
Jelaskan fungsi dari penggunaan await
ketika kita menggunakan fetch()
! Apa yang akan terjadi jika kita tidak menggunakan await
?
Penggunaan await
dalam konteks asynchronous JavaScript berfungsi untuk menunggu hasil dari promise yang dikembalikan oleh fungsi fetch()
, sehingga memungkinkan kode untuk dieksekusi dengan cara yang lebih mudah dibaca dan dikelola. Ketika await
digunakan, kode akan berhenti sejenak sampai promise diselesaikan, memastikan bahwa data yang diperlukan sudah tersedia sebelum melanjutkan ke baris berikutnya. Jika await
tidak digunakan, fetch akan mengembalikan promise tanpa menunggu, yang menyebabkan eksekusi kode berlanjut sebelum response diterima, sehingga dapat mengakibatkan kesalahan saat mencoba mengakses data yang belum ada.
Decorator csrf_exempt
digunakan untuk menonaktifkan pemeriksaan CSRF (Cross-Site Request Forgery) pada view yang menerima permintaan AJAX POST, memberikan kenyamanan dalam pengembangan tanpa mengorbankan keamanan yang berlebihan. CSRF adalah serangan yang berusaha memanipulasi pengguna untuk melakukan tindakan yang tidak diinginkan, namun dalam kasus permintaan dari frontend yang aman (misalnya, dari pengguna yang terautentikasi), penggunaan csrf_exempt
dapat mempermudah implementasi. Dengan menonaktifkan pemeriksaan CSRF untuk view tertentu, pengembang dapat fokus pada logika bisnis tanpa harus mengelola kompleksitas validasi yang seringkali terjadi pada permintaan AJAX.
Pada tutorial PBP minggu ini, pembersihan data input pengguna dilakukan di belakang (backend) juga. Mengapa hal tersebut tidak dilakukan di frontend saja?
Pembersihan data input pengguna dilakukan di backend karena alasan keamanan, konsistensi data, dan pengendalian logika bisnis. Data yang diterima dari frontend dapat dimanipulasi, sehingga validasi dan pembersihan di backend memastikan bahwa data yang diproses dan disimpan di database telah diverifikasi dan aman dari ancaman seperti injeksi SQL dan XSS. Selain itu, pembersihan di backend menjaga konsistensi data sesuai dengan aturan bisnis yang ditetapkan, dan sering kali memerlukan logika pembersihan yang kompleks yang lebih baik dikelola di backend, memungkinkan pemisahan tanggung jawab dan meningkatkan pemeliharaan kode.
Saya memulai dengan mengubah kode yang mengelola data mood agar mendukung AJAX GET
. Saya memastikan bahwa hanya data milik pengguna yang terautentikasi yang diambil. Langkah pertama adalah membuka views.py
dan menghapus kode yang mengambil objek mood entry berdasarkan pengguna. Saya kemudian memperbarui fungsi show_json
dan show_xml
menjadi:
data = Product.objects.filter(user=request.user)
Selanjutnya, saya beralih ke profile/index.html
, di mana saya menghapus bagian conditional yang menampilkan pesan saat tidak ada data produk. Sebagai pengganti, saya menambahkan elemen berikut:
<div id="product_cards"></div>
Di bagian <script>
, saya membuat fungsi getProducts()
untuk mengambil data produk dengan menggunakan fetch API:
async function getProducts() {
return fetch("/json").then((res) => res.json())
}
Saya juga menambahkan fungsi refreshProducts()
untuk memperbarui tampilan data produk secara dinamis.
Saya menambahkan modal menggunakan Tailwind CSS untuk aplikasi saya. Modal ini ditempatkan di bawah div dengan ID product_cards
. Untuk mengaktifkan modal, saya menambahkan fungsi JavaScript:
function showModal() {
modal.classList.remove('hidden');
setTimeout(() => modalContent.classList.remove('opacity-0', 'scale-95'), 50);
}
function hideModal() {
modalContent.classList.add('opacity-0', 'scale-95');
setTimeout(() => modal.classList.add('hidden'), 150);
}
document.getElementById("cancelButton").addEventListener("click", hideModal);
document.getElementById("closeModalBtn").addEventListener("click", hideModal);
Saya juga mengganti tombol "Create Product" dengan tombol untuk membuka modal:
<button onclick="showModal();">Create Product by AJAX</button>
Selanjutnya, saya membuat fungsi addProduct
untuk mengirim data melalui AJAX:
function addProduct() {
fetch("/create-ajax", {
method: "POST",
body: new FormData(document.querySelector('#productForm')),
}).then(response => refreshProducts())
document.getElementById("productForm").reset();
hideModal();
return false;
}
document.getElementById("submitProductForm").onclick = addProduct