diff --git a/week06/lab6.html b/week06/lab6.html new file mode 100644 index 0000000..fa299f3 --- /dev/null +++ b/week06/lab6.html @@ -0,0 +1,548 @@ + + + + + + + +Lab 6 Amirlan Sharipov (BS-CS21-01) + + + + + +
+

Lab 6 Amirlan Sharipov (BS-CS21-01)

+
+

Table of Contents

+ +
+ +
+

1. Disclaimer

+
+

+Please, use the lab6.sql file to read/copy the source code. Also, the html version of this document looks better than the pdf one. +

+
+
+ +
+

2. Exercise 1

+
+
+
+

2.1. Table creation and insertion

+
+

+I used the schema provided in the slides. And then manually inserted data into the tables. +

+ +
+
CREATE TABLE customers (
+    customerId INT,
+    customerName VARCHAR(50),
+    city VARCHAR(50),
+    PRIMARY KEY (customerId)
+);
+
+CREATE TABLE items (
+    itemId INT,
+    itemName VARCHAR(50),
+    price FLOAT,
+    PRIMARY KEY (itemId)
+);
+
+CREATE TABLE orders (
+    orderId INT,
+    customerId INT,
+    date DATE,
+    FOREIGN KEY (customerId) REFERENCES customers(customerId),
+    PRIMARY KEY (orderId)
+);
+
+CREATE TABLE order_details (
+    orderId INT,
+    itemId INT,
+    quantity INT,
+    FOREIGN KEY (itemId) REFERENCES items(itemId),
+    PRIMARY KEY (orderId, itemId)
+);
+
+INSERT INTO customers VALUES('101', 'Martin', 'Prague');
+INSERT INTO customers VALUES('107', 'Herman', 'Madrid');
+INSERT INTO customers VALUES('110', 'Pedro', 'Moscow');
+
+INSERT INTO items VALUES('3786', 'Net', 35.0);
+INSERT INTO items VALUES('4011', 'Racket', 65.0);
+INSERT INTO items VALUES('9132', 'Pack-3', 4.75);
+INSERT INTO items VALUES('5794', 'Pack-6', 5.0);
+INSERT INTO items VALUES('3141', 'Cover', 10.0);
+
+INSERT INTO orders VALUES('2301', '101', '2011-02-23');
+INSERT INTO orders VALUES('2302', '107', '2011-02-25');
+INSERT INTO orders VALUES('2303', '110', '2011-02-27');
+
+INSERT INTO order_details VALUES ('2301', '3786', 3);
+INSERT INTO order_details VALUES ('2301', '4011', 6);
+INSERT INTO order_details VALUES ('2301', '9132', 8);
+INSERT INTO order_details VALUES ('2302', '5794', 4);
+INSERT INTO order_details VALUES ('2303', '4011', 2);
+INSERT INTO order_details VALUES ('2303', '3141', 2);
+
+
+
+
+ +
+

2.2. Queries

+
+

+First query takes the sum of all prices * quantities groupped by the orders and sorts them. +The second query does the same thing, groups by customers, sorts by sum of quantities (descending order) and takes the first result. +

+
+
SELECT order_details.orderId, SUM(items.price * order_details.quantity)
+FROM order_details
+INNER JOIN items on items.itemId=order_details.itemId
+group by order_details.orderId ORDER BY sum ASC;
+
+SELECT customers.customerName, customers.city FROM customers
+INNER JOIN orders ON customers.customerId=orders.customerId
+INNER JOIN order_details ON order_details.orderId=orders.orderId
+GROUP BY customers.customerId
+ORDER BY SUM(order_details.quantity) DESC
+LIMIT 1;
+
+
+
+
+
+ +
+

3. Exercise 2

+
+
+
+

3.1. Normalization

+
+

+I have several assumptions: +

+
    +
  • Any teacher can work at several schools at once (or change the school)
  • +
  • Room numbers don’t depend on schools: the first number of the room name is not enough to assume the opposite
  • +
  • Teachers may teach several courses: it’s an elementary school, usually teachers can teach anything in elementary schools
  • +
+
+ +
+

3.1.1. 1NF

+
+

+It’s almost in 1NF state. Each cell is already atomic, values of the same domain, etc. The only thing that’s not there is the primary key. Let’s say for now that the primary key is a tuple of school, teacher, course, room, grade, and book. This way there are no conflicts. It still looks like a mess, so I will normalize it further. +

+
+
+
+

3.1.2. 2NF

+
+

+Make new tables with relations for partial functional dependencies of non-prime attributes on candidate keys: +

+
    +
  • lessons (lessonId, schoolName, teacherName, courseName, roomName, gradeName)
  • +
  • books (bookId, bookName, publisherName)
  • +
  • loans (loanId, lessonId, bookId, loanDate)
  • +
+
+
+
+

3.1.3. 3NF

+
+

+Make new tables (with appropriate IDs) with relations for transitive functional dependencies of non-prime attribute on candidate key: +

+
    +
  • schools (schoolId, schoolName)
  • +
  • teachers (teacherId, teacherName)
  • +
  • courses (courseId, courseName)
  • +
  • rooms (roomId, roomName)
  • +
  • grades (gradeId, gradeName)
  • +
  • publishers (publisherId, publisherName)
  • +
  • lessons (lessonId, schoolId, teacherId, courseId, roomId, gradeId)
  • +
  • books (bookId, bookName, publisherId)
  • +
  • loans (loanId, lessonId, bookId, loanDate)
  • +
+
+
+
+

3.1.4. BCNF and 4NF

+
+

+Already satisfies. +

+
+
+ +
+

3.1.5. Code

+
+
+
CREATE TABLE schools (
+    schoolId SERIAL,
+    schoolName VARCHAR(50),
+    PRIMARY KEY (schoolId)
+);
+
+CREATE TABLE teachers (
+    teacherId SERIAL,
+    teacherName VARCHAR(30),
+    PRIMARY KEY (teacherId)
+);
+
+CREATE TABLE courses (
+    courseId SERIAL,
+    courseName VARCHAR(40),
+    PRIMARY KEY (courseId)
+);
+
+CREATE TABLE rooms (
+    roomId SERIAL,
+    roomName VARCHAR(40),
+    PRIMARY KEY (roomId)
+);
+
+CREATE TABLE grades (
+    gradeId SERIAL,
+    gradeName VARCHAR(15),
+    PRIMARY KEY (gradeId)
+);
+
+CREATE TABLE publishers (
+    publisherId SERIAL,
+    publisherName VARCHAR(30),
+    PRIMARY KEY (publisherId)
+);
+
+CREATE TABLE books (
+    bookId SERIAL,
+    bookName VARCHAR(60),
+    publisherId INT,
+    FOREIGN KEY (publisherId) REFERENCES publishers(publisherId),
+    PRIMARY KEY (bookId)
+);
+
+CREATE TABLE lessons (
+    lessonId SERIAL,
+    schoolId INT,
+    teacherId INT,
+    courseId INT,
+    roomId INT,
+    gradeId INT,
+    FOREIGN KEY (teacherId) REFERENCES teachers(teacherId),
+    FOREIGN KEY (courseId) REFERENCES courses(courseId),
+    FOREIGN KEY (roomId) REFERENCES rooms(roomId),
+    FOREIGN KEY (gradeId) REFERENCES grades(gradeId),
+    PRIMARY KEY (lessonId)
+);
+
+CREATE TABLE loans (
+    loanId SERIAL,
+    lessonId INT,
+    bookId INT,
+    loanDate DATE,
+    FOREIGN KEY (lessonId) REFERENCES lessons(lessonId),
+    FOREIGN KEY (bookId) REFERENCES books(bookId),
+    PRIMARY KEY (loanId)
+);
+
+INSERT INTO schools (schoolName)
+    SELECT DISTINCT school FROM loan_books;
+INSERT INTO teachers (teacherName)
+    SELECT DISTINCT teacher FROM loan_books;
+-- Inserted Numerical thinking 2 times because of case sensitivity. Not gonna change anything.
+INSERT INTO courses (courseName)
+    SELECT DISTINCT course FROM loan_books;
+INSERT INTO rooms (roomName)
+    SELECT DISTINCT room FROM loan_books;
+INSERT INTO grades (gradeName)
+    SELECT DISTINCT grade FROM loan_books;
+INSERT INTO publishers (publisherName)
+    SELECT DISTINCT publisher FROM loan_books;
+INSERT INTO books (bookName, publisherId)
+    SELECT DISTINCT loan_books.book,
+    publishers.publisherId FROM loan_books
+    INNER JOIN publishers ON
+        publishers.publisherName=loan_books.publisher;
+
+INSERT INTO lessons (schoolId, teacherId, courseId, roomId, gradeId)
+    SELECT DISTINCT schools.schoolId, teachers.teacherId,
+        courses.courseId, rooms.roomId, grades.gradeId FROM loan_books
+    INNER JOIN schools ON schools.schoolName=loan_books.school
+    INNER JOIN teachers ON teachers.teacherName=loan_books.teacher
+    INNER JOIN courses ON courses.courseName=loan_books.course
+    INNER JOIN rooms ON rooms.roomName=loan_books.room
+    INNER JOIN grades ON grades.gradeName=loan_books.grade;
+
+INSERT INTO loans (lessonId, bookId, loanDate)
+    SELECT DISTINCT lessons.lessonId, books.bookId,
+        loan_books.loanDate FROM loan_books
+    INNER JOIN schools ON schools.schoolName=loan_books.school
+    INNER JOIN teachers ON teachers.teacherName=loan_books.teacher
+    INNER JOIN courses ON courses.courseName=loan_books.course
+    INNER JOIN rooms ON rooms.roomName=loan_books.room
+    INNER JOIN grades ON grades.gradeName=loan_books.grade
+    INNER JOIN books ON books.bookName=loan_books.book
+    INNER JOIN lessons ON lessons.gradeId=grades.gradeId
+        and lessons.roomId=rooms.roomId
+        and lessons.courseId=courses.courseId
+        and lessons.teacherId=teachers.teacherId;
+
+
+
+
+
+
+

3.2. Queries

+
+

+The first query list all the schools that borrowed the books of every publisher using DISTINCT keyword. +The second query orders the results from each school, and takes only 1 loan that has the highest loanDate for each school. +

+
+
SELECT DISTINCT publishers.publisherName, books.bookName,
+    schools.schoolName FROM books
+    INNER JOIN loans ON loans.bookId=books.bookId
+    INNER JOIN lessons ON lessons.lessonId=loans.lessonId
+    INNER JOIN schools ON schools.schoolId=lessons.schoolId
+    INNER JOIN publishers ON publishers.publisherId=books.publisherId
+    ORDER BY publishers.publisherName;
+
+SELECT DISTINCT ON (schools.schoolName) schools.schoolName,
+    publishers.publisherName, books.bookName FROM loans
+    INNER JOIN lessons ON lessons.lessonId=loans.lessonId
+    INNER JOIN schools ON schools.schoolId=lessons.schoolId
+    INNER JOIN books ON books.bookId=loans.bookId
+    INNER JOIN publishers ON publishers.publisherId=books.publisherId
+    ORDER BY schools.schoolName, loans.loanDate DESC, 1;
+
+
+
+
+
+
+
+

Author: Amirlan Sharipov (BS-CS21-01)

+

Created: 2023-02-28 Tue 23:48

+
+ + \ No newline at end of file diff --git a/week06/lab6.org b/week06/lab6.org new file mode 100644 index 0000000..be599bd --- /dev/null +++ b/week06/lab6.org @@ -0,0 +1,237 @@ +#+title: Lab 6 Amirlan Sharipov (BS-CS21-01) +#+author: Amirlan Sharipov (BS-CS21-01) + +* Exercise 1 +** Table creation and insertion +I used the schema provided in the slides. And then manually inserted data into the tables. + +#+begin_src sql +CREATE TABLE customers ( + customerId INT, + customerName VARCHAR(50), + city VARCHAR(50), + PRIMARY KEY (customerId) +); + +CREATE TABLE items ( + itemId INT, + itemName VARCHAR(50), + price FLOAT, + PRIMARY KEY (itemId) +); + +CREATE TABLE orders ( + orderId INT, + customerId INT, + date DATE, + FOREIGN KEY (customerId) REFERENCES customers(customerId), + PRIMARY KEY (orderId) +); + +CREATE TABLE order_details ( + orderId INT, + itemId INT, + quantity INT, + FOREIGN KEY (itemId) REFERENCES items(itemId), + PRIMARY KEY (orderId, itemId) +); + +INSERT INTO customers VALUES('101', 'Martin', 'Prague'); +INSERT INTO customers VALUES('107', 'Herman', 'Madrid'); +INSERT INTO customers VALUES('110', 'Pedro', 'Moscow'); + +INSERT INTO items VALUES('3786', 'Net', 35.0); +INSERT INTO items VALUES('4011', 'Racket', 65.0); +INSERT INTO items VALUES('9132', 'Pack-3', 4.75); +INSERT INTO items VALUES('5794', 'Pack-6', 5.0); +INSERT INTO items VALUES('3141', 'Cover', 10.0); + +INSERT INTO orders VALUES('2301', '101', '2011-02-23'); +INSERT INTO orders VALUES('2302', '107', '2011-02-25'); +INSERT INTO orders VALUES('2303', '110', '2011-02-27'); + +INSERT INTO order_details VALUES ('2301', '3786', 3); +INSERT INTO order_details VALUES ('2301', '4011', 6); +INSERT INTO order_details VALUES ('2301', '9132', 8); +INSERT INTO order_details VALUES ('2302', '5794', 4); +INSERT INTO order_details VALUES ('2303', '4011', 2); +INSERT INTO order_details VALUES ('2303', '3141', 2); +#+end_src + +** Queries +First query takes the sum of all prices * quantities groupped by the orders and sorts them. +The second query does the same thing, groups by customers, sorts by sum of quantities (descending order) and takes the first result. +#+begin_src sql +SELECT order_details.orderId, SUM(items.price * order_details.quantity) +FROM order_details +INNER JOIN items on items.itemId=order_details.itemId +group by order_details.orderId ORDER BY sum ASC; + +SELECT customers.customerName, customers.city FROM customers +INNER JOIN orders ON customers.customerId=orders.customerId +INNER JOIN order_details ON order_details.orderId=orders.orderId +GROUP BY customers.customerId +ORDER BY SUM(order_details.quantity) DESC +LIMIT 1; +#+end_src + +* Exercise 2 +** Normalization +I have several assumptions: +- Any teacher can work at several schools at once (or change the school) +- Room numbers don't depend on schools: the first number of the room name is not enough to assume the opposite +- Teachers may teach several courses: it's an elementary school, usually teachers can teach anything in elementary schools + +*** 1NF +It's almost in 1NF state. Each cell is already atomic, values of the same domain, etc. The only thing that's not there is the primary key. Let's say for now that the primary key is a tuple of school, teacher, course, room, grade, and book. This way there are no conflicts. It still looks like a mess, so I will normalize it further. +*** 2NF +Make new tables with relations for partial functional dependencies of non-prime attributes on candidate keys: +- lessons (lessonId, schoolName, teacherName, courseName, roomName, gradeName) +- books (bookId, bookName, publisherName) +- loans (loanId, lessonId, bookId, loanDate) +*** 3NF +Make new tables (with appropriate IDs) with relations for transitive functional dependencies of non-prime attribute on candidate key: +- schools (schoolId, schoolName) +- teachers (teacherId, teacherName) +- courses (courseId, courseName) +- rooms (roomId, roomName) +- grades (gradeId, gradeName) +- publishers (publisherId, publisherName) +- lessons (lessonId, schoolId, teacherId, courseId, roomId, gradeId) +- books (bookId, bookName, publisherId) +- loans (loanId, lessonId, bookId, loanDate) +*** BCNF and 4NF +Already satisfies. + +*** Code +#+begin_src sql +CREATE TABLE schools ( + schoolId SERIAL, + schoolName VARCHAR(50), + PRIMARY KEY (schoolId) +); + +CREATE TABLE teachers ( + teacherId SERIAL, + teacherName VARCHAR(30), + PRIMARY KEY (teacherId) +); + +CREATE TABLE courses ( + courseId SERIAL, + courseName VARCHAR(40), + PRIMARY KEY (courseId) +); + +CREATE TABLE rooms ( + roomId SERIAL, + roomName VARCHAR(40), + PRIMARY KEY (roomId) +); + +CREATE TABLE grades ( + gradeId SERIAL, + gradeName VARCHAR(15), + PRIMARY KEY (gradeId) +); + +CREATE TABLE publishers ( + publisherId SERIAL, + publisherName VARCHAR(30), + PRIMARY KEY (publisherId) +); + +CREATE TABLE books ( + bookId SERIAL, + bookName VARCHAR(60), + publisherId INT, + FOREIGN KEY (publisherId) REFERENCES publishers(publisherId), + PRIMARY KEY (bookId) +); + +CREATE TABLE lessons ( + lessonId SERIAL, + schoolId INT, + teacherId INT, + courseId INT, + roomId INT, + gradeId INT, + FOREIGN KEY (teacherId) REFERENCES teachers(teacherId), + FOREIGN KEY (courseId) REFERENCES courses(courseId), + FOREIGN KEY (roomId) REFERENCES rooms(roomId), + FOREIGN KEY (gradeId) REFERENCES grades(gradeId), + PRIMARY KEY (lessonId) +); + +CREATE TABLE loans ( + loanId SERIAL, + lessonId INT, + bookId INT, + loanDate DATE, + FOREIGN KEY (lessonId) REFERENCES lessons(lessonId), + FOREIGN KEY (bookId) REFERENCES books(bookId), + PRIMARY KEY (loanId) +); + +INSERT INTO schools (schoolName) + SELECT DISTINCT school FROM loan_books; +INSERT INTO teachers (teacherName) + SELECT DISTINCT teacher FROM loan_books; +-- Inserted Numerical thinking 2 times because of case sensitivity. Not gonna change anything. +INSERT INTO courses (courseName) + SELECT DISTINCT course FROM loan_books; +INSERT INTO rooms (roomName) + SELECT DISTINCT room FROM loan_books; +INSERT INTO grades (gradeName) + SELECT DISTINCT grade FROM loan_books; +INSERT INTO publishers (publisherName) + SELECT DISTINCT publisher FROM loan_books; +INSERT INTO books (bookName, publisherId) + SELECT DISTINCT loan_books.book, + publishers.publisherId FROM loan_books + INNER JOIN publishers ON + publishers.publisherName=loan_books.publisher; + +INSERT INTO lessons (schoolId, teacherId, courseId, roomId, gradeId) + SELECT DISTINCT schools.schoolId, teachers.teacherId, + courses.courseId, rooms.roomId, grades.gradeId FROM loan_books + INNER JOIN schools ON schools.schoolName=loan_books.school + INNER JOIN teachers ON teachers.teacherName=loan_books.teacher + INNER JOIN courses ON courses.courseName=loan_books.course + INNER JOIN rooms ON rooms.roomName=loan_books.room + INNER JOIN grades ON grades.gradeName=loan_books.grade; + +INSERT INTO loans (lessonId, bookId, loanDate) + SELECT DISTINCT lessons.lessonId, books.bookId, + loan_books.loanDate FROM loan_books + INNER JOIN schools ON schools.schoolName=loan_books.school + INNER JOIN teachers ON teachers.teacherName=loan_books.teacher + INNER JOIN courses ON courses.courseName=loan_books.course + INNER JOIN rooms ON rooms.roomName=loan_books.room + INNER JOIN grades ON grades.gradeName=loan_books.grade + INNER JOIN books ON books.bookName=loan_books.book + INNER JOIN lessons ON lessons.gradeId=grades.gradeId + and lessons.roomId=rooms.roomId + and lessons.courseId=courses.courseId + and lessons.teacherId=teachers.teacherId; +#+end_src +** Queries +The first query list all the schools that borrowed the books of every publisher using DISTINCT keyword. +The second query orders the results from each school, and takes only 1 loan that has the highest loanDate for each school. +#+begin_src sql +SELECT DISTINCT publishers.publisherName, books.bookName, + schools.schoolName FROM books + INNER JOIN loans ON loans.bookId=books.bookId + INNER JOIN lessons ON lessons.lessonId=loans.lessonId + INNER JOIN schools ON schools.schoolId=lessons.schoolId + INNER JOIN publishers ON publishers.publisherId=books.publisherId + ORDER BY publishers.publisherName; + +SELECT DISTINCT ON (schools.schoolName) schools.schoolName, + publishers.publisherName, books.bookName FROM loans + INNER JOIN lessons ON lessons.lessonId=loans.lessonId + INNER JOIN schools ON schools.schoolId=lessons.schoolId + INNER JOIN books ON books.bookId=loans.bookId + INNER JOIN publishers ON publishers.publisherId=books.publisherId + ORDER BY schools.schoolName, loans.loanDate DESC, 1; +#+end_src diff --git a/week06/lab6.pdf b/week06/lab6.pdf new file mode 100644 index 0000000..55027ea Binary files /dev/null and b/week06/lab6.pdf differ diff --git a/week06/lab6.sql b/week06/lab6.sql new file mode 100644 index 0000000..7d9a811 --- /dev/null +++ b/week06/lab6.sql @@ -0,0 +1,197 @@ +-- EXERCISE 1 +-- I inserted everything manually + +CREATE TABLE customers ( + customerId INT, + customerName VARCHAR(50), + city VARCHAR(50), + PRIMARY KEY (customerId) +); + +CREATE TABLE items ( + itemId INT, + itemName VARCHAR(50), + price FLOAT, + PRIMARY KEY (itemId) +); + +CREATE TABLE orders ( + orderId INT, + customerId INT, + date DATE, + FOREIGN KEY (customerId) REFERENCES customers(customerId), + PRIMARY KEY (orderId) +); + +CREATE TABLE order_details ( + orderId INT, + itemId INT, + quantity INT, + FOREIGN KEY (itemId) REFERENCES items(itemId), + PRIMARY KEY (orderId, itemId) +); + +INSERT INTO customers VALUES('101', 'Martin', 'Prague'); +INSERT INTO customers VALUES('107', 'Herman', 'Madrid'); +INSERT INTO customers VALUES('110', 'Pedro', 'Moscow'); + +INSERT INTO items VALUES('3786', 'Net', 35.0); +INSERT INTO items VALUES('4011', 'Racket', 65.0); +INSERT INTO items VALUES('9132', 'Pack-3', 4.75); +INSERT INTO items VALUES('5794', 'Pack-6', 5.0); +INSERT INTO items VALUES('3141', 'Cover', 10.0); + +INSERT INTO orders VALUES('2301', '101', '2011-02-23'); +INSERT INTO orders VALUES('2302', '107', '2011-02-25'); +INSERT INTO orders VALUES('2303', '110', '2011-02-27'); + +INSERT INTO order_details VALUES ('2301', '3786', 3); +INSERT INTO order_details VALUES ('2301', '4011', 6); +INSERT INTO order_details VALUES ('2301', '9132', 8); +INSERT INTO order_details VALUES ('2302', '5794', 4); +INSERT INTO order_details VALUES ('2303', '4011', 2); +INSERT INTO order_details VALUES ('2303', '3141', 2); + +SELECT order_details.orderId, SUM(items.price * order_details.quantity) +FROM order_details +INNER JOIN items on items.itemId=order_details.itemId +group by order_details.orderId ORDER BY sum ASC; + + +SELECT customers.customerName, customers.city FROM customers +INNER JOIN orders ON customers.customerId=orders.customerId +INNER JOIN order_details ON order_details.orderId=orders.orderId +GROUP BY customers.customerId +ORDER BY SUM(order_details.quantity) DESC +LIMIT 1; + +-- Exercise 2 +-- Automated insert + +CREATE TABLE loan_books + (school VARCHAR(50), + teacher VARCHAR(30), + course VARCHAR(40), + room VARCHAR(10), + grade VARCHAR(15), + book VARCHAR(60), + publisher VARCHAR(30), + loanDate DATE, + PRIMARY KEY (school, teacher, course) + ); + +INSERT INTO loan_books VALUES ('Horizon Education Institute', 'Chad Russell', 'Logical Thinking', '1.A01', '1st grade', 'Learning and teaching in early childhood education', 'BOA Editions', '2010-09-09'); +INSERT INTO loan_books VALUES ('Horizon Education Institute', 'Chad Russell', 'Writing', '1.A01', '1st grade', 'Preschool, N56', 'Taylor & Francis Publishing', '2010-05-05'); +INSERT INTO loan_books VALUES ('Horizon Education Institute', 'Chad Russell', 'Numerical thinking', '1.A01', '1st grade', 'Learning and teaching in early childhood education', 'BOA Editions', '2010-05-05'); +INSERT INTO loan_books VALUES ('Horizon Education Institute', 'E.F.Codd', 'Spatial, Temporal and Causal Thinking', '1.B01', '1st grade', 'Early Childhood Education N9', 'Prentice Hall', '2010-05-06'); +INSERT INTO loan_books VALUES ('Horizon Education Institute', 'E.F.Codd', 'Numerical thinking', '1.B01', '1st grade', 'Learning and teaching in early childhood education', 'BOA Editions', '2010-05-06'); +INSERT INTO loan_books VALUES ('Horizon Education Institute', 'Jones Smith', 'Writing', '1.A01', '2nd grade', 'Learning and teaching in early childhood education', 'BOA Editions', '2010-09-09'); +INSERT INTO loan_books VALUES ('Horizon Education Institute', 'Jones Smith', 'English', '1.A01', '2nd grade', 'Know how to educate: guide for Parents and Teachers', 'McGraw Hill', '2010-05-05'); +INSERT INTO loan_books VALUES ('Bright Institution', 'Adam Baker', 'Logical Thinking', '2.B01', '1st grade', 'Know how to educate: guide for Parents and Teachers', 'McGraw Hill', '2010-12-18'); +INSERT INTO loan_books VALUES ('Bright Institution', 'Adam Baker', 'Numerical Thinking', '2.B01', '1st grade', 'Learning and teaching in early childhood education', 'BOA Editions', '2010-05-06'); + +CREATE TABLE schools ( + schoolId SERIAL, + schoolName VARCHAR(50), + PRIMARY KEY (schoolId) +); + +CREATE TABLE teachers ( + teacherId SERIAL, + teacherName VARCHAR(30), + PRIMARY KEY (teacherId) +); + +CREATE TABLE courses ( + courseId SERIAL, + courseName VARCHAR(40), + PRIMARY KEY (courseId) +); + +CREATE TABLE rooms ( + roomId SERIAL, + roomName VARCHAR(40), + PRIMARY KEY (roomId) +); + +CREATE TABLE grades ( + gradeId SERIAL, + gradeName VARCHAR(15), + PRIMARY KEY (gradeId) +); + +CREATE TABLE publishers ( + publisherId SERIAL, + publisherName VARCHAR(30), + PRIMARY KEY (publisherId) +); + +CREATE TABLE books ( + bookId SERIAL, + bookName VARCHAR(60), + publisherId INT, + FOREIGN KEY (publisherId) REFERENCES publishers(publisherId), + PRIMARY KEY (bookId) +); + +CREATE TABLE lessons ( + lessonId SERIAL, + schoolId INT, + teacherId INT, + courseId INT, + roomId INT, + gradeId INT, + FOREIGN KEY (teacherId) REFERENCES teachers(teacherId), + FOREIGN KEY (courseId) REFERENCES courses(courseId), + FOREIGN KEY (roomId) REFERENCES rooms(roomId), + FOREIGN KEY (gradeId) REFERENCES grades(gradeId), + PRIMARY KEY (lessonId) +); + +CREATE TABLE loans ( + loanId SERIAL, + lessonId INT, + bookId INT, + loanDate DATE, + FOREIGN KEY (lessonId) REFERENCES lessons(lessonId), + FOREIGN KEY (bookId) REFERENCES books(bookId), + PRIMARY KEY (loanId) +); + +INSERT INTO schools (schoolName) SELECT DISTINCT school FROM loan_books; +INSERT INTO teachers (teacherName) SELECT DISTINCT teacher FROM loan_books; +INSERT INTO courses (courseName) SELECT DISTINCT course FROM loan_books; -- Inserted Numerical thinking 2 times because of case sensitivity. Not gonna change anything. +INSERT INTO rooms (roomName) SELECT DISTINCT room FROM loan_books; +INSERT INTO grades (gradeName) SELECT DISTINCT grade FROM loan_books; +INSERT INTO publishers (publisherName) SELECT DISTINCT publisher FROM loan_books; +INSERT INTO books (bookName, publisherId) SELECT DISTINCT loan_books.book, publishers.publisherId FROM loan_books INNER JOIN publishers ON publishers.publisherName=loan_books.publisher; + +INSERT INTO lessons (schoolId, teacherId, courseId, roomId, gradeId) SELECT DISTINCT schools.schoolId, teachers.teacherId, courses.courseId, rooms.roomId, grades.gradeId FROM loan_books + INNER JOIN schools ON schools.schoolName=loan_books.school + INNER JOIN teachers ON teachers.teacherName=loan_books.teacher + INNER JOIN courses ON courses.courseName=loan_books.course + INNER JOIN rooms ON rooms.roomName=loan_books.room + INNER JOIN grades ON grades.gradeName=loan_books.grade; + +INSERT INTO loans (lessonId, bookId, loanDate) SELECT DISTINCT lessons.lessonId, books.bookId, loan_books.loanDate FROM loan_books + INNER JOIN schools ON schools.schoolName=loan_books.school + INNER JOIN teachers ON teachers.teacherName=loan_books.teacher + INNER JOIN courses ON courses.courseName=loan_books.course + INNER JOIN rooms ON rooms.roomName=loan_books.room + INNER JOIN grades ON grades.gradeName=loan_books.grade + INNER JOIN books ON books.bookName=loan_books.book + INNER JOIN lessons ON lessons.gradeId=grades.gradeId and lessons.roomId=rooms.roomId and lessons.courseId=courses.courseId and lessons.teacherId=teachers.teacherId; + +SELECT DISTINCT publishers.publisherName, books.bookName, schools.schoolName FROM books + INNER JOIN loans ON loans.bookId=books.bookId + INNER JOIN lessons ON lessons.lessonId=loans.lessonId + INNER JOIN schools ON schools.schoolId=lessons.schoolId + INNER JOIN publishers ON publishers.publisherId=books.publisherId + ORDER BY publishers.publisherName; + +SELECT DISTINCT ON (schools.schoolName) schools.schoolName, publishers.publisherName, books.bookName, loans.loanDate FROM loans + INNER JOIN lessons ON lessons.lessonId=loans.lessonId + INNER JOIN schools ON schools.schoolId=lessons.schoolId + INNER JOIN books ON books.bookId=loans.bookId + INNER JOIN publishers ON publishers.publisherId=books.publisherId + ORDER BY schools.schoolName, loans.loanDate DESC, 1; diff --git a/week06/lab6.tex b/week06/lab6.tex new file mode 100644 index 0000000..8e724d0 --- /dev/null +++ b/week06/lab6.tex @@ -0,0 +1,286 @@ +% Created 2023-02-28 Tue 23:48 +% Intended LaTeX compiler: pdflatex +\documentclass[11pt]{article} +\usepackage[utf8]{inputenc} +\usepackage[T1]{fontenc} +\usepackage{graphicx} +\usepackage{longtable} +\usepackage{wrapfig} +\usepackage{rotating} +\usepackage[normalem]{ulem} +\usepackage{amsmath} +\usepackage{amssymb} +\usepackage{capt-of} +\usepackage{hyperref} +\author{Amirlan Sharipov (BS-CS21-01)} +\date{\today} +\title{Lab 6 Amirlan Sharipov (BS-CS21-01)} +\hypersetup{ + pdfauthor={Amirlan Sharipov (BS-CS21-01)}, + pdftitle={Lab 6 Amirlan Sharipov (BS-CS21-01)}, + pdfkeywords={}, + pdfsubject={}, + pdfcreator={Emacs 28.2 (Org mode 9.6)}, + pdflang={English}} +\begin{document} + +\maketitle +\tableofcontents + + +\section{Disclaimer} +\label{sec:orgd9c7526} +Please, use the lab6.sql file to read/copy the source code. Also, the html version of this document looks better than the pdf one. + +\section{Exercise 1} +\label{sec:orgbb19d8e} +\subsection{Table creation and insertion} +\label{sec:orge2ed6dc} +I used the schema provided in the slides. And then manually inserted data into the tables. + +\begin{verbatim} +CREATE TABLE customers ( + customerId INT, + customerName VARCHAR(50), + city VARCHAR(50), + PRIMARY KEY (customerId) +); + +CREATE TABLE items ( + itemId INT, + itemName VARCHAR(50), + price FLOAT, + PRIMARY KEY (itemId) +); + +CREATE TABLE orders ( + orderId INT, + customerId INT, + date DATE, + FOREIGN KEY (customerId) REFERENCES customers(customerId), + PRIMARY KEY (orderId) +); + +CREATE TABLE order_details ( + orderId INT, + itemId INT, + quantity INT, + FOREIGN KEY (itemId) REFERENCES items(itemId), + PRIMARY KEY (orderId, itemId) +); + +INSERT INTO customers VALUES('101', 'Martin', 'Prague'); +INSERT INTO customers VALUES('107', 'Herman', 'Madrid'); +INSERT INTO customers VALUES('110', 'Pedro', 'Moscow'); + +INSERT INTO items VALUES('3786', 'Net', 35.0); +INSERT INTO items VALUES('4011', 'Racket', 65.0); +INSERT INTO items VALUES('9132', 'Pack-3', 4.75); +INSERT INTO items VALUES('5794', 'Pack-6', 5.0); +INSERT INTO items VALUES('3141', 'Cover', 10.0); + +INSERT INTO orders VALUES('2301', '101', '2011-02-23'); +INSERT INTO orders VALUES('2302', '107', '2011-02-25'); +INSERT INTO orders VALUES('2303', '110', '2011-02-27'); + +INSERT INTO order_details VALUES ('2301', '3786', 3); +INSERT INTO order_details VALUES ('2301', '4011', 6); +INSERT INTO order_details VALUES ('2301', '9132', 8); +INSERT INTO order_details VALUES ('2302', '5794', 4); +INSERT INTO order_details VALUES ('2303', '4011', 2); +INSERT INTO order_details VALUES ('2303', '3141', 2); +\end{verbatim} + +\subsection{Queries} +\label{sec:org4f468b3} +First query takes the sum of all prices * quantities groupped by the orders and sorts them. +The second query does the same thing, groups by customers, sorts by sum of quantities (descending order) and takes the first result. +\begin{verbatim} +SELECT order_details.orderId, SUM(items.price * order_details.quantity) +FROM order_details +INNER JOIN items on items.itemId=order_details.itemId +group by order_details.orderId ORDER BY sum ASC; + +SELECT customers.customerName, customers.city FROM customers +INNER JOIN orders ON customers.customerId=orders.customerId +INNER JOIN order_details ON order_details.orderId=orders.orderId +GROUP BY customers.customerId +ORDER BY SUM(order_details.quantity) DESC +LIMIT 1; +\end{verbatim} + +\section{Exercise 2} +\label{sec:org99a84d3} +\subsection{Normalization} +\label{sec:org77cd4b4} +I have several assumptions: +\begin{itemize} +\item Any teacher can work at several schools at once (or change the school) +\item Room numbers don't depend on schools: the first number of the room name is not enough to assume the opposite +\item Teachers may teach several courses: it's an elementary school, usually teachers can teach anything in elementary schools +\end{itemize} + +\subsubsection{1NF} +\label{sec:org4242f0d} +It's almost in 1NF state. Each cell is already atomic, values of the same domain, etc. The only thing that's not there is the primary key. Let's say for now that the primary key is a tuple of school, teacher, course, room, grade, and book. This way there are no conflicts. It still looks like a mess, so I will normalize it further. +\subsubsection{2NF} +\label{sec:orge350be8} +Make new tables with relations for partial functional dependencies of non-prime attributes on candidate keys: +\begin{itemize} +\item lessons (lessonId, schoolName, teacherName, courseName, roomName, gradeName) +\item books (bookId, bookName, publisherName) +\item loans (loanId, lessonId, bookId, loanDate) +\end{itemize} +\subsubsection{3NF} +\label{sec:org627722d} +Make new tables (with appropriate IDs) with relations for transitive functional dependencies of non-prime attribute on candidate key: +\begin{itemize} +\item schools (schoolId, schoolName) +\item teachers (teacherId, teacherName) +\item courses (courseId, courseName) +\item rooms (roomId, roomName) +\item grades (gradeId, gradeName) +\item publishers (publisherId, publisherName) +\item lessons (lessonId, schoolId, teacherId, courseId, roomId, gradeId) +\item books (bookId, bookName, publisherId) +\item loans (loanId, lessonId, bookId, loanDate) +\end{itemize} +\subsubsection{BCNF and 4NF} +\label{sec:org00e5b2a} +Already satisfies. + +\subsubsection{Code} +\label{sec:org3de05cf} +\begin{verbatim} +CREATE TABLE schools ( + schoolId SERIAL, + schoolName VARCHAR(50), + PRIMARY KEY (schoolId) +); + +CREATE TABLE teachers ( + teacherId SERIAL, + teacherName VARCHAR(30), + PRIMARY KEY (teacherId) +); + +CREATE TABLE courses ( + courseId SERIAL, + courseName VARCHAR(40), + PRIMARY KEY (courseId) +); + +CREATE TABLE rooms ( + roomId SERIAL, + roomName VARCHAR(40), + PRIMARY KEY (roomId) +); + +CREATE TABLE grades ( + gradeId SERIAL, + gradeName VARCHAR(15), + PRIMARY KEY (gradeId) +); + +CREATE TABLE publishers ( + publisherId SERIAL, + publisherName VARCHAR(30), + PRIMARY KEY (publisherId) +); + +CREATE TABLE books ( + bookId SERIAL, + bookName VARCHAR(60), + publisherId INT, + FOREIGN KEY (publisherId) REFERENCES publishers(publisherId), + PRIMARY KEY (bookId) +); + +CREATE TABLE lessons ( + lessonId SERIAL, + schoolId INT, + teacherId INT, + courseId INT, + roomId INT, + gradeId INT, + FOREIGN KEY (teacherId) REFERENCES teachers(teacherId), + FOREIGN KEY (courseId) REFERENCES courses(courseId), + FOREIGN KEY (roomId) REFERENCES rooms(roomId), + FOREIGN KEY (gradeId) REFERENCES grades(gradeId), + PRIMARY KEY (lessonId) +); + +CREATE TABLE loans ( + loanId SERIAL, + lessonId INT, + bookId INT, + loanDate DATE, + FOREIGN KEY (lessonId) REFERENCES lessons(lessonId), + FOREIGN KEY (bookId) REFERENCES books(bookId), + PRIMARY KEY (loanId) +); + +INSERT INTO schools (schoolName) + SELECT DISTINCT school FROM loan_books; +INSERT INTO teachers (teacherName) + SELECT DISTINCT teacher FROM loan_books; +-- Inserted Numerical thinking 2 times because of case sensitivity. Not gonna change anything. +INSERT INTO courses (courseName) + SELECT DISTINCT course FROM loan_books; +INSERT INTO rooms (roomName) + SELECT DISTINCT room FROM loan_books; +INSERT INTO grades (gradeName) + SELECT DISTINCT grade FROM loan_books; +INSERT INTO publishers (publisherName) + SELECT DISTINCT publisher FROM loan_books; +INSERT INTO books (bookName, publisherId) + SELECT DISTINCT loan_books.book, + publishers.publisherId FROM loan_books + INNER JOIN publishers ON + publishers.publisherName=loan_books.publisher; + +INSERT INTO lessons (schoolId, teacherId, courseId, roomId, gradeId) + SELECT DISTINCT schools.schoolId, teachers.teacherId, + courses.courseId, rooms.roomId, grades.gradeId FROM loan_books + INNER JOIN schools ON schools.schoolName=loan_books.school + INNER JOIN teachers ON teachers.teacherName=loan_books.teacher + INNER JOIN courses ON courses.courseName=loan_books.course + INNER JOIN rooms ON rooms.roomName=loan_books.room + INNER JOIN grades ON grades.gradeName=loan_books.grade; + +INSERT INTO loans (lessonId, bookId, loanDate) + SELECT DISTINCT lessons.lessonId, books.bookId, + loan_books.loanDate FROM loan_books + INNER JOIN schools ON schools.schoolName=loan_books.school + INNER JOIN teachers ON teachers.teacherName=loan_books.teacher + INNER JOIN courses ON courses.courseName=loan_books.course + INNER JOIN rooms ON rooms.roomName=loan_books.room + INNER JOIN grades ON grades.gradeName=loan_books.grade + INNER JOIN books ON books.bookName=loan_books.book + INNER JOIN lessons ON lessons.gradeId=grades.gradeId + and lessons.roomId=rooms.roomId + and lessons.courseId=courses.courseId + and lessons.teacherId=teachers.teacherId; +\end{verbatim} +\subsection{Queries} +\label{sec:org917674e} +The first query list all the schools that borrowed the books of every publisher using DISTINCT keyword. +The second query orders the results from each school, and takes only 1 loan that has the highest loanDate for each school. +\begin{verbatim} +SELECT DISTINCT publishers.publisherName, books.bookName, + schools.schoolName FROM books + INNER JOIN loans ON loans.bookId=books.bookId + INNER JOIN lessons ON lessons.lessonId=loans.lessonId + INNER JOIN schools ON schools.schoolId=lessons.schoolId + INNER JOIN publishers ON publishers.publisherId=books.publisherId + ORDER BY publishers.publisherName; + +SELECT DISTINCT ON (schools.schoolName) schools.schoolName, + publishers.publisherName, books.bookName FROM loans + INNER JOIN lessons ON lessons.lessonId=loans.lessonId + INNER JOIN schools ON schools.schoolId=lessons.schoolId + INNER JOIN books ON books.bookId=loans.bookId + INNER JOIN publishers ON publishers.publisherId=books.publisherId + ORDER BY schools.schoolName, loans.loanDate DESC, 1; +\end{verbatim} +\end{document} \ No newline at end of file