From 57730b9399696607b89fc7d2549f5d67594e46ca Mon Sep 17 00:00:00 2001 From: Sorrel Harriet <s.harriet@gold.ac.uk> Date: Fri, 4 Nov 2016 14:19:12 +0000 Subject: [PATCH] initial work on lab 5 example code --- lab-5/music-store-app/artist.php | 35 ++++++ lab-5/music-store-app/index.php | 45 ++++++++ lab-5/music-store-app/sql/dummy_data.sql | 66 +++++++++++ lab-5/music-store-app/sql/queries.sql | 100 ++++++++++++++++ lab-5/music-store-app/sql/schema.sql | 138 +++++++++++++++++++++++ 5 files changed, 384 insertions(+) create mode 100755 lab-5/music-store-app/artist.php create mode 100755 lab-5/music-store-app/index.php create mode 100755 lab-5/music-store-app/sql/dummy_data.sql create mode 100755 lab-5/music-store-app/sql/queries.sql create mode 100755 lab-5/music-store-app/sql/schema.sql diff --git a/lab-5/music-store-app/artist.php b/lab-5/music-store-app/artist.php new file mode 100755 index 0000000..f50a022 --- /dev/null +++ b/lab-5/music-store-app/artist.php @@ -0,0 +1,35 @@ +<?php + +/* connect to the database */ +$link = mysqli_connect( + 'localhost', + 'YOUR_USERNAME', + 'YOUR_PASSWORD', + 'YOUR_DB_NAME' +); + +/* check connection succeeded */ +if (mysqli_connect_errno()) { + exit(mysqli_connect_error()); +} + +/* fetch a result set */ +$sql = "SELECT first_name, last_name FROM artist"; +$result = mysqli_query($link, $sql); + +/* check query returned a result */ +if ($result === false) { + echo mysqli_error($link); +} else { + /* fetch associative array */ + while ($row = mysqli_fetch_assoc($result)) { + echo($row['first_name']." (".$row['last_name'].") "); + } + /* free result set */ + mysqli_free_result($result); +} + +/* close the connection to the database */ +mysqli_close($link) + +?> \ No newline at end of file diff --git a/lab-5/music-store-app/index.php b/lab-5/music-store-app/index.php new file mode 100755 index 0000000..de97e70 --- /dev/null +++ b/lab-5/music-store-app/index.php @@ -0,0 +1,45 @@ +<?php + +// open a new MySQL database connection + +// define a connection 'handle' +$link = mysqli_connect( + 'localhost', + 'YOUR_USERNAME', + 'YOUR_PASSWORD', + 'YOUR_DB_NAME' +); + +// check connection succeeded +if (mysqli_connect_errno()) { + exit(mysqli_connect_error()); +} + +// define the SQL query to run (from lab 3!) +$sql = "SELECT Album.title, Album.price, Artist.first_name, Artist.last_name, Genre.name AS genre, (SELECT COUNT(*) FROM Track WHERE Album.upc=Track.album_upc) AS num_tracks FROM Album +LEFT JOIN Track ON Album.upc=Track.album_upc +INNER JOIN Artist + ON Album.artist_id=Artist.id +INNER JOIN Genre + ON Album.genre_id=Genre.id +ORDER BY title ASC"; +$result = mysqli_query($link, $sql); + +// check query returned a result +if ($result === false) { + echo mysqli_error($link); +} else { + // $row_cnt = mysqli_num_rows($result); + // echo $row_cnt + // fetch associative array */ + while ($row = mysqli_fetch_assoc($result)) { + echo($row['title']." (".$row['genre'].") "); + } + // free result set + mysqli_free_result($result); +} + +// close the connection to the database */ +mysqli_close($link) + +?> diff --git a/lab-5/music-store-app/sql/dummy_data.sql b/lab-5/music-store-app/sql/dummy_data.sql new file mode 100755 index 0000000..382b373 --- /dev/null +++ b/lab-5/music-store-app/sql/dummy_data.sql @@ -0,0 +1,66 @@ +INSERT INTO Genre (name) +VALUES +('Reggae'), +('Funk and Soul'), +('Jazz'), +('Classical'), +('Electronic'), +('Pop'); + +INSERT INTO Band (name, genre_id) +VALUES +('The Wailers', 1), +('The Aces', 2), +('The Beatles', 6); + +INSERT INTO Artist (first_name, last_name, band_id) +VALUES +('Bob', 'Marley', 1), +('Peter', 'Tosh', 1), +('Burning', 'Spear', NULL), +('Alton', 'Ellis', NULL), +('Gregory', 'Issacs', NULL), +('Desmond', 'Dekker', 2), +('John', 'Lennon', 3), +('Madonna', NULL, NULL); + +INSERT INTO Album (upc, title, release_year, artist_id, compilation, price, genre_id) +VALUES +('123425364732', 'Soul Rebel', 1970, 1, FALSE, 25.99, 1 ), +('017263547261', 'Catch A Fire', 1973, 1, 0, 25.99, 1 ), +('018263526272', 'Natty Dread', 1974, 1, 1, 20.99, 1 ), +('018273527273', 'Babylon By Bus', 1978, 1, 'FALSE', 24.99, 1 ), +('491827364626', 'Night Nurse', 1982, 5, TRUE, 17.99, 1 ), +('018276272828', 'Mr Issacs', 1982, 5, 0, 9.99, 1 ), +('018273662728', 'Black and Dekker', 1980, 6, 0, 19.99, 1 ), +('726517237627', 'Sunday Coming', 1970, 4, TRUE, 15.99, 1 ), +('018274372722', 'Imagine', 1971, 7, NULL, 11.99, 6 ), +('018273727287', 'Like a Virgin', 1984, 8, 1, 9.99, 6 ); + +INSERT INTO Track (name, album_upc, track_number) +VALUES +('Sunday Coming', '726517237627', 1), +('Imagine', '018274372722', 1), +('Material Girl', '018273727287', 1), +('Angel', '018273727287', 2), +('Crippled Inside', '018274372722', 2), +('These Eyes', '726517237627', 2), +('Hurting Me', '726517237627', 3); + +INSERT INTO Customer (username, email_address, password, first_name, last_name, address_1, address_2, postcode) +VALUES +('sharr003', 's.harriet@gold.ac.uk', 'SOME_HASHED_VALUE', 'Sorrel', 'Harriet', '12 Fake Street', 'London', 'SE140PL'), +('ktack001', 'k.tackie@hotmail.com', 'SOME_HASHED_VALUE', 'Kobi', 'Tackie', '12 Fake Street', 'London', 'SE140PL'); + +INSERT INTO Transaction (customer_id, delivery) +VALUES +(1, 'next working day'), +(1, 'first class'), +(2, 'second class'); + +INSERT INTO LineItem (trans_id, album_upc, quantity) +VALUES +(1, '018273727287', 1), +(2, '018273727287', 1), +(3, '491827364626', 1), +(1, '018274372722', 2); diff --git a/lab-5/music-store-app/sql/queries.sql b/lab-5/music-store-app/sql/queries.sql new file mode 100755 index 0000000..50ec236 --- /dev/null +++ b/lab-5/music-store-app/sql/queries.sql @@ -0,0 +1,100 @@ +/* Minimum functional requirements of +Music Store App and their associated queries. +Where variations are given they are +given in order of increasing complexity */ + +/* 1a. Retrieve a list of Albums +Include the UPC, title, price, artist name +and genre in the result-set. +Order by title (a-z) */ +SELECT Album.title, Album.price, Artist.first_name, Artist.last_name, Genre.name AS genre +FROM Album /* notice the use of a column alias! */ +INNER JOIN Artist + ON Album.artist_id=Artist.id +INNER JOIN Genre + ON Album.genre_id=Genre.id +ORDER BY title ASC; /* causes results to be ordered by album title */ + +/* 1b. Retrieve a list of Albums +Include the UPC, title, price, artist name, +genre, and number of tracks in the result-set. +Include only albums which have tracks in DB. +Order by title (a-z) */ +SELECT Album.title, Album.price, Artist.first_name, Artist.last_name, Genre.name AS genre, (SELECT COUNT(*) FROM Track WHERE Album.upc=Track.album_upc) AS num_tracks +FROM Album /* notice the nested query, aggregate function and aliases! */ +INNER JOIN Track + ON Album.upc=Track.album_upc +INNER JOIN Artist + ON Album.artist_id=Artist.id +INNER JOIN Genre + ON Album.genre_id=Genre.id +ORDER BY title ASC; + +/* 1c. Retrieve a list of Albums +Include the UPC, title, price, artist name, +genre, and number of tracks in the result-set. +Include ALL albums, even those without tracks. +Order by title (a-z) */ +SELECT Album.title, Album.price, Artist.first_name, Artist.last_name, Genre.name AS genre, (SELECT COUNT(*) FROM Track WHERE Album.upc=Track.album_upc) AS num_tracks +FROM Album /* notice the nested query, aggregate function and aliases! */ +LEFT JOIN Track /* LEFT JOIN causes all Albums to be returned, even when no matching record in Track table */ + ON Album.upc=Track.album_upc +INNER JOIN Artist + ON Album.artist_id=Artist.id +INNER JOIN Genre + ON Album.genre_id=Genre.id +ORDER BY title ASC; + +/* ------------------------------------ */ + +/* 2. Retrieve a list of tracks +associated with a specific album */ +SELECT t.name AS track, t.track_number, a.title AS album /* notice use of column aliases! */ +FROM Track t /* notice use of table aliases! */ +INNER JOIN Album a + ON t.album_upc=a.upc +WHERE a.upc='018274372722' /* EXAMPLE */ +ORDER BY track_number ASC; + +/* ------------------------------------ */ + +/* 3. Insert a new track */ +INSERT INTO Track (name, album_upc, track_number) +VALUES +('Sunday Coming', '726517237627', 1); /* EXAMPLE */ + +/* ------------------------------------ */ + +/* 4. Delete a track */ +DELETE FROM Track WHERE album_upc='726517237627'; /* EXAMPLE */ + +/* ------------------------------------ */ + +/* 5a. Retrieve details about an order +Include the customer_id, delivery method +and total order value in the result-set */ + +SELECT Transaction.customer_id, Transaction.delivery, SUM(Album.price*LineItem.quantity) AS order_total /* notice use of aggregate function! */ +FROM Transaction +INNER JOIN LineItem + ON Transaction.id=LineItem.trans_id +INNER JOIN Album + ON Album.upc=LineItem.album_upc +WHERE Transaction.id=1; /* EXAMPLE */ + +/* 5b. Retrieve details about an order +Include the transaction id, customer_id +and the number of items ordered in the +result-set */ +SELECT t.id, t.customer_id, SUM(li.quantity) AS num_items +FROM Transaction t +INNER JOIN LineItem li + ON t.id=li.trans_id +GROUP BY t.id; /* group results by transaction id */ + +/* 5c. Does the same as the previous query +but using a nested query rather than a JOIN */ +SELECT t.id, t.customer_id, +(SELECT SUM(li.quantity) FROM LineItem li + WHERE li.trans_id=t.id) AS num_items +FROM Transaction t; diff --git a/lab-5/music-store-app/sql/schema.sql b/lab-5/music-store-app/sql/schema.sql new file mode 100755 index 0000000..8d8f95a --- /dev/null +++ b/lab-5/music-store-app/sql/schema.sql @@ -0,0 +1,138 @@ +/* Delete existing tables in reverse order of creation +so as not to violate any foreign key constraints */ +DROP TABLE IF EXISTS Rating, Wish, LineItem, Transaction, Customer, Track, Album, Artist, Band, Genre; + +/* Define table for genres */ +CREATE TABLE Genre ( + id INT AUTO_INCREMENT, + name VARCHAR(25) UNIQUE NOT NULL, + PRIMARY KEY (id) +); + +/* Define table for storing act (i.e. a group or solo artist) +ON DELETE CASCADE not used in this case as we don't want bands +being deleted if their genre is deleted. We'd rather an error +was raised. */ +CREATE TABLE Band ( + id INT AUTO_INCREMENT, + name VARCHAR(50) NOT NULL, + genre_id INT, + PRIMARY KEY (id), + FOREIGN KEY (genre_id) + REFERENCES Genre (id) +); + +/* Define table for storing artists */ +CREATE TABLE Artist ( + id INT AUTO_INCREMENT, + first_name VARCHAR(25) NOT NULL, + last_name VARCHAR(25), + band_id INT, + PRIMARY KEY (id), + FOREIGN KEY (band_id) + REFERENCES Band (id) +); + +/* Define table for storing albums */ +CREATE TABLE Album ( + upc CHAR(12), + title VARCHAR(50) NOT NULL, + release_year YEAR(4), + artist_id INT, + compilation BOOLEAN NOT NULL DEFAULT 0, + price DECIMAL(5, 2) unsigned NOT NULL, + genre_id INT, + PRIMARY KEY (upc), + FOREIGN KEY (artist_id) + REFERENCES Artist (id), + FOREIGN KEY (genre_id) + REFERENCES Genre (id) +); + + +/* Define table for storing single tracks +ON DELETE CASCADE will ensure no track +is left that isn't linked to an album. +ON UPDATE CASCADE will inherit any changes +to the Album UPC (Universal Product Code). +A composite PRIMARY KEY is used here to +ensure the track_number is unique +on any given album */ +CREATE TABLE Track ( + name VARCHAR(50) NOT NULL, + album_upc CHAR(12) NOT NULL, + track_number INT, + PRIMARY KEY (album_upc, track_number), + FOREIGN KEY (album_upc) + REFERENCES Album (upc) + ON UPDATE CASCADE + ON DELETE CASCADE +); + +/* Define table for storing customers */ +CREATE TABLE Customer ( + id INT AUTO_INCREMENT, + username VARCHAR(25) UNIQUE NOT NULL, + email_address VARCHAR(50) UNIQUE NOT NULL, + password VARCHAR(64), /* will store a hashed password */ + first_name VARCHAR(50) NOT NULL, + last_name VARCHAR(50) NOT NULL, + address_1 VARCHAR(50) NOT NULL, + address_2 VARCHAR(50), + postcode VARCHAR(10) NOT NULL, + PRIMARY KEY (id) +); + +/* Define table for storing orders */ +CREATE TABLE Transaction ( + id INT AUTO_INCREMENT, + customer_id INT NOT NULL, + delivery ENUM('first class','second class','next working day'), + PRIMARY KEY (id), + FOREIGN KEY (customer_id) + REFERENCES Customer(id) + ON DELETE CASCADE +); + +/* Define table for storing line items */ +CREATE TABLE LineItem ( + id INT AUTO_INCREMENT, + trans_id INT NOT NULL, + album_upc CHAR(12) NOT NULL, + quantity INT NOT NULL DEFAULT 1, + PRIMARY KEY (id), + FOREIGN KEY (trans_id) + REFERENCES Transaction (id) + ON DELETE CASCADE, + FOREIGN KEY (album_upc) + REFERENCES Album (upc) + ON DELETE CASCADE +); + +/* Define table to store customer 'wishes'. +Compile wishlists from this data. */ +CREATE TABLE Wish ( + customer_id INT, + album_upc CHAR(12) NOT NULL, + PRIMARY KEY (customer_id, album_upc), + FOREIGN KEY (customer_id) + REFERENCES Customer(id) + ON DELETE CASCADE, + FOREIGN KEY (album_upc) + REFERENCES Album (upc) + ON DELETE CASCADE +); + +/* Define table for storing customer ratingsfor albums */ +CREATE TABLE Rating ( + customer_id INT NOT NULL, + album_upc CHAR(12) NOT NULL, + score TINYINT unsigned NOT NULL, + PRIMARY KEY (customer_id, album_upc), + FOREIGN KEY (customer_id) + REFERENCES Customer(id) + ON DELETE CASCADE, + FOREIGN KEY (album_upc) + REFERENCES Album (upc) + ON DELETE CASCADE +); -- GitLab