linking child tables of two related parent-child tables - sql

I have two parent child tables and a couple of supporting tables. I am struggling to figure out the right relationships between them.
Orders, a table that holds a list of Orders including:Order Id, Supplier, Order Date etc..
OrderDetails, a Table holds a list of products that have been ordered, and is a child of Orders, linked on OrderId. Key fields are the ProductId and the Quantity Ordered. Each order containes one or many OrderDetails that outline the products and quantities being ordered.
Shipments, a table that holds a list of Shipments, including Tracking Number and Shipper. Links to Orders through Order Id. An Order may have multiple shipments. For example, I ordered 100 lightbulbs. They were dispatched in 5 shipments.
ShipmentDetails, holds a list of product and shipped quantities and is linked to the Shipments table by ShippingId. Each Shipment may have multiple ShipmentDetails. i.e. One shipment may have 30 Lightbulbs and 10 Door Knobs.
Products is a table that holds a list of Products that need to be both ordered and shipped.
So the logic is that I enter details about the products that are to be ordered in Products. For example, 100 CREE LED Lightbulbs, and 50 Door Handles.
When I place an order, I create an Order in Orders.
i.e. amazon.com, order #45454.
Then I add child rows to that order in OrderDetails. i.e. 30 CREE LED Lightbulbs.
When the order Ships, I create an entry in the Shipments table. i.e. Tracking #46464646464, linked to OrderId in Orders. And then I enter what is in that shipment in ShipmentDetails. For example, only 20 or the 30 CREE LED Lightbulbs may be associated with this Shipping entry.
I am struggling with figuring out how to relate the Shipping Detail records to the Order Detail Table. Lets say the Shipping Detail Table has 4 fields.
ShippingID - a link to the Shipments table parent.
TrackingNum - a field that holds a tracking number.
Product - this should be a drop down, that says, the shipment I belong to is related to a defined Order (because ShipmentDetail is a child of a Shipment record which in turn holds a key to the Orders table, which in turn has OrderDetails that reference products).
Quantity - this should have a default (overridable) value that is the "Quantity Ordered" number from the OrdersDetail record that matches the Order Id and Product Id. Where OrderId is in the Shipments table (linked to the Orders table) and ProductID comes from #3 above. The reason it must be overridable is that a shipment may be a partial shipment.
I am stuck with the preceding #3 and #4. I hope I have explained this in a vaguely understandable way! The pictures below may help!
The ask:
What is the correct join between the ShipmentDetails and the OrderDetail Table
How do I create a field in the ShipmentDetail table with a default value pulled from the Quantities Field of the OrderDetail table, where
Shipments!OrderId = Orders!Id and ShipmentDetail!ProductID = OrderDetails!Product ID
I am working in MS Access 2016 - but I suspect this is a fairly generic SQL question...rather than MS Access specific.

I think the naming is confusing.
What I would rather have is (each first id is autoincrement, forgot how to say this on access):
// Product doesn't contain any information about quantity/price etc ...
create table Products(product_id int, name text, description text ...);
// Orders:
create table Orders(order_id int, client_name text, description text ...);
create table OrderDetails(order_detail_id int, order_id int, product_id int, quantity double, unit_price double, ...);
// Shipments
create table Shipments(shipment_id int, company text, description text ...);
create table ShipmentDetails(shipment_detail_id int, shipment_id int, product_id int, quantity double, price double default 0, ...);
// inserting shipments per product (defaulting to total quantity per product), assuming we have shipment_id SID
insert into ShipmentDetails(shipment_id, order_id, product_id, quantity)
select SID, order_id, product_id, SUM(quantity)
from OrderDetails
group by order_id, product_id;
Then you can of course have some filters (date, customer etc ...).
For the first question, I am not clear what exaclty you want to return.
Here is a comparison of quantities:
select t.order_id, t.product_id, sum(t.quantity) as product_quantity, sum(u.quantity) as shipment_quantity
from OrderDetails t inner join ShipmentDetails u
on t.order_id = u.order_id and t.product_id = u.product_id;

Related

getting the sum of products between two columns in SQL

My situation is this: I have an order_details table containing entities representing the
"content" of an orders, each row has
a product_id column ---> foreign key referencing the type of product
an order_id column ----> foreign key represents which order the product belongs to
a qty column ----------> representing the quantity of the product present in the order
each product row instead has:
a product name column ----> representing the product name
a co2_value --------------> representing the value of co2 removed buying that product
each table (orders table too) has his own id column as primary key for the rows.
the problem is the following:
I need a sql query to find the total co2_value of an order adding all the co2_values of the products belonging to a specific order, multiplied by their quantity.
I tried this query first:
SELECT SUM(co2_value) FROM products
INNER JOIN order_details
ON products.id = order_details.product_id AND order_details.order_id = ". $this->id;
the problem with this is that i'am getting only the sum of the co2_values of the product without multiplying them by the quantity, is there a way to do that in a single query?
I'll leave the link to migration here
select
sum(co2_value * qty)
from
products
inner join order_details on
products.id = order_details.product_id
and order_details.order_id = ". $this->id;
You can simply select the co2_value with respect to its quantity by multiplying it using the above query

How would I structure a database for taking sales orders?

I am helping someone build a grocery deliver service. Very simple site and order process. The problem I am having is knowing how to come up with schema for the orders. The order would have contact information, but would also need to have all of the products they ordered. Since it can/will be different for each persons order, how would I go about designing this? Would I just have a products ordered field with a string list of all the items? Or would I need multiple linked tables?
Thanks!
You need three tables:
a) Products - Contains the product details
b) Orders - Contains the order header, contact information etc
c) Product_Orders - is a table with two columns: product_id and order_id, that bind the orders to the products, and a quantity field and unit price.
You could have a customers table, but ideally information like the delivery address, etc, would be attached to the order so that if the user changes his address it will not affect your order history.
For the same reason unit price must be in Product_Orders, so that if a product price changes it does not affect previous orders.
Product table
with name and price of all products.
Customer table
with name, address, phone etc...
Order table
with entries for each order: order id, cusomer id, date etc
Order Items table
with all ordered items, with link to id in ordertable, product table, quantity, price, etc...

SAP HANA: Customer history

I have different tables with customer Id, items bought with a date of purchase.
books table consists of customer id, bookNr and date of purchase.
shoes table consists of custoemr id, shoeNr and date of purchase.
.
.
we have like for every item a table.
If the customer buys an Item(x) lets say a book with bookNr someNumber. I would like to retrieve all the orders in all the tables before the date of purchase of bookNr someNumber for all the customers.
Any suggestions how to design the query and the best way to represent the data.
The way you designed the tables mixes two concepts (type of item that got purchased and the purchase itself). This leads to the difficulties you now face.
An alternative approach would be to have a PURCHASES table that links to an "item" table. In the "item" table you could differentiate what item it is and, depending on how different the items are, link to further detail tables, that are specific to the item type.
This could be a DB design that looks like this:
[PURCHASES] -- [ITEMS] -- [BOOK_ITEM_DETAILS]
|
+-------[SHOE_ITEM_DETAILS]
If you want to keep your current design, the probably easiest approach here would be to create a view that combines (unions) the different item tables.
E.g.
CREATE view CUSTOMER_PURCHASES as
SELECT customer_id, shoe_Nr as item_nr, date_of_purchase FROM shoes
UNION ALL
SELECT customer_id, book_Nr as item_nr, date_of_purchase FROM books
...
That way, all purchases are available in a single view.
Now, if you look for all items purchased by a user before that same user bought a specific item (e.g. a shoe), you can use a join query.
SELECT customer_id, item_nr, date_of_purchase
FROM customer_purchases cp
left outer join
(SELECT customer_id, item_nr, date_of_purchase
FROM customer_purchases
where item_nr = <shoe_item_nr>
and date_of_purchase = <date_of_purchase>
and customer_id = <customer_id>) as ref_purchase
on
cp.customer_id = ref_purchase.customer_id
and cp.date_of_purchase < ref_purchase.date_of_purchase;

Simple database for product order

I want to make a simple database to order products, like chips/drinks (as full product, without any specific info about product just name and price for unit)
I designed this but I'm not sure if it's good:
**Person:**
id
username
name
password
phone
email
street
zip
**order:**
id
person_id
product_id
date
quantity (neccessary?)
status (done or not)
**product:**
id
name
price
relations:
[person] 1 --- 1 [order] 1 --- many [product]
(I'm not sure about relations and fields)
It seems that in your way you are going to end up in orders containing a single product (even if you use the quantity)
I would modify the Order table:
**order:**
id
person_id
date
status (done or not)
And I would add a new table:
**OrderDetails**
id
order_id
product_id
quantity
You may check out for db normalization. You should add columns to a table that are directly related to the table. For instance date in the order is valid, because it refers to the order it was made. On the other hand it wouldn't be valid in the person table (unless it was referring to the person join date). So, similarly the quantity refers to the product in the order (thus in OrderDetails) not in the Order or the Product.
You will probably need an intermediate table between order and product, so you can add many times same order to different products

How to set the value of a column based on other column

I have the following schema:
customer(cid,cname),
books(isbn,price)
orders(oid,cid,isbn,odate,quantity,value)
1. A customer can purchase any no.of books on a single order on the same date
2. A customer can purchase any no. of different types on a single orders
3. A customer can place multiple orders on a same day
So, when a customer place order for isbn say 121,141 whose prices are 500,500 respectively, of quantity 2,5 respectively, the column 'value' must be filled with 1*500=500 and 5*500=2500 respectively.
So the value column must be quantity*price always. How can I specify this during the creation of orders table? Should I need a trigger?
I think there is a problem with your model: your orders table refers to the books table via the isbn field, when an order can contain several books !
I would add an intermediate table:
single_order(sid, oid, isbn, quantity)
referring to the orders one via the oid field. An item of orders would then be the sum of the items of single_order having the corresponding oid. And the value field would then be the sum of the price of books by the quantity of single_order, you can easily link to these 2 tables with joins.