| | 1 | == Определување функционални зависности |
| | 2 | |
| | 3 | R = {user_id, first_name, last_name, phone, email, password, reservation_id, start_date, end_date, reserv_date, status, building_id, address, city, floors, num_rooms, room_number, room_type, number_of_beds, price_per_night, available, service_id, service_type, service_date, service_status, p_id, p_method, amount, p_date} |
| | 4 | |
| | 5 | === Функционални зависности |
| | 6 | |
| | 7 | Users |
| | 8 | user_id → first_name, last_name, phone, email, password |
| | 9 | |
| | 10 | Customer (extends User) |
| | 11 | user_id → {customer role} |
| | 12 | |
| | 13 | Manager (extends User) |
| | 14 | user_id → {manager role} |
| | 15 | |
| | 16 | Staff (extends User) |
| | 17 | user_id → {staff role} |
| | 18 | |
| | 19 | Hotel Building |
| | 20 | building_id → address, city, floors, num_rooms |
| | 21 | |
| | 22 | Room (слаб ентитет) |
| | 23 | (building_id, room_number) → room_type, number_of_beds, price_per_night, available |
| | 24 | |
| | 25 | Reservation |
| | 26 | reservation_id → start_date, end_date, reserv_date, status, user_id (customer), manager_id, room_number, building_id |
| | 27 | |
| | 28 | Payment |
| | 29 | p_id → p_method, amount, p_date, reservation_id, customer_id |
| | 30 | |
| | 31 | Service |
| | 32 | service_id → service_type, service_date, service_status, staff_id, room_number, building_id |
| | 33 | |
| | 34 | === Класификација на атрибути |
| | 35 | |
| | 36 | Лево – атрибути кои одредуваат други \\ |
| | 37 | user_id, building_id, (building_id, room_number), reservation_id, service_id, p_id |
| | 38 | |
| | 39 | Лево и десно – атрибути кои се и детерминанти и зависни \\ |
| | 40 | room_number (во комбинација со building_id), manager_id, staff_id, customer_id |
| | 41 | |
| | 42 | Десно – атрибути кои се само зависни \\ |
| | 43 | first_name, last_name, phone, email, password, address, city, floors, num_rooms, room_type, number_of_beds, price_per_night, available, start_date, end_date, reserv_date, status, p_method, amount, p_date, service_type, service_date, service_status |
| | 44 | |
| | 45 | === Покривачи на примарни клучеви |
| | 46 | |
| | 47 | User |
| | 48 | user_id+ = {user_id, first_name, last_name, phone, email, password} |
| | 49 | |
| | 50 | Hotel Building |
| | 51 | building_id+ = {building_id, address, city, floors, num_rooms} |
| | 52 | |
| | 53 | Room |
| | 54 | (building_id, room_number)+ = {room_number, building_id, room_type, number_of_beds, price_per_night, available, address, city, floors, num_rooms} |
| | 55 | |
| | 56 | Reservation |
| | 57 | reservation_id+ = {reservation_id, start_date, end_date, reserv_date, status, user_id (customer), manager_id, room_number, building_id} |
| | 58 | |
| | 59 | Payment |
| | 60 | p_id+ = {p_id, p_method, amount, p_date, reservation_id, customer_id} |
| | 61 | |
| | 62 | Service |
| | 63 | service_id+ = {service_id, service_type, service_date, service_status, staff_id, room_number, building_id} |
| | 64 | |
| | 65 | Следува дека { user_id, building_id, room_number, reservation_id, payment_id, service_id } е единствен кандидат клуч и го прогласуваме за примарен клуч. |
| | 66 | |
| | 67 | === Анализа според покривачи |
| | 68 | |
| | 69 | Од анализата се забележуваат: |
| | 70 | |
| | 71 | room_number е уникатен само во комбинација со building_id → слаб ентитет. |
| | 72 | |
| | 73 | Во Reservation нема експлицитно модел на врска со Customer, Manager и Room → треба релации за интегритет. |
| | 74 | |
| | 75 | Во Payment, amount е транзитивно зависно од Reservation (amount = цена * број на ноќи). |
| | 76 | |
| | 77 | Во Service, моментално нема детална историја (само статус). |
| | 78 | |
| | 79 | === Анализа на нормални форми |
| | 80 | 1 НФ |
| | 81 | |
| | 82 | Тековниот дизајн е во 1NF: атомски атрибути, нема повеќевредносни атрибути складирани како едно поле. |
| | 83 | |
| | 84 | 2 НФ |
| | 85 | |
| | 86 | Нема парцијални зависности (освен кај Room, каде клуч е составен (building_id, room_number)). |
| | 87 | |
| | 88 | 3 НФ |
| | 89 | |
| | 90 | Проблеми: |
| | 91 | |
| | 92 | Reservation – amount (треба да се изведува од Room.price_per_night × број на ноќи → не треба да стои во Payment како атрибут). |
| | 93 | |
| | 94 | Reservation моментално чува и customer_id, и manager_id → транзитивна зависност преку User. |
| | 95 | |
| | 96 | Подобрувања за да се задоволи 3 НФ |
| | 97 | Reservation – Order Cost |
| | 98 | |
| | 99 | Да не се чува amount во Payment, туку да се пресметува. |
| | 100 | |
| | 101 | |
| | 102 | ALTER TABLE Payment DROP COLUMN amount; |
| | 103 | |
| | 104 | Да се додава VIEW: |
| | 105 | |
| | 106 | CREATE VIEW reservation_total AS |
| | 107 | SELECT r.reservation_id, |
| | 108 | (EXTRACT(DAY FROM r.end_date - r.start_date) * rm.price_per_night) AS total_cost |
| | 109 | FROM Reservation r |
| | 110 | JOIN Room rm ON r.room_number = rm.room_number AND r.building_id = rm.building_id; |
| | 111 | |
| | 112 | Service – историја на статуси |
| | 113 | |
| | 114 | CREATE TABLE service_history ( |
| | 115 | history_id BIGINT GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY, |
| | 116 | service_id BIGINT NOT NULL, |
| | 117 | status VARCHAR(70) NOT NULL, |
| | 118 | changed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, |
| | 119 | FOREIGN KEY (service_id) REFERENCES Service(service_id) |
| | 120 | ); |
| | 121 | |
| | 122 | BCNF |
| | 123 | Прекршоци |
| | 124 | |
| | 125 | email во Hotel_User треба да е уникатен → веќе има UNIQUE. |
| | 126 | |
| | 127 | phone исто така треба да е уникатен (еден број = еден корисник). |
| | 128 | |
| | 129 | Во Room, комбинацијата (building_id, room_number) е единствен идентификатор. |
| | 130 | |
| | 131 | ALTER TABLE Hotel_User ADD CONSTRAINT uq_phone UNIQUE(phone); |
| | 132 | |
| | 133 | ALTER TABLE Room ADD CONSTRAINT uq_room UNIQUE(building_id, room_number); |