Skip to content

Commit 90f631c

Browse files
committedMar 17, 2025
feat: extract ct data into csv command
1 parent 7a27355 commit 90f631c

File tree

1 file changed

+139
-0
lines changed

1 file changed

+139
-0
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,139 @@
1+
from datetime import datetime
2+
import csv
3+
from enum import Enum
4+
from itertools import product
5+
6+
from commerce_coordinator.apps.commercetools.management.commands._ct_api_client_command import (
7+
CommercetoolsAPIClientCommand
8+
)
9+
10+
11+
# Enum for product types
12+
class ProductType(Enum):
13+
EDX_COURSE_ENTITLEMENT = "edx_course_entitlement"
14+
EDX_PROGRAM = "edx_program"
15+
OC_SELF_PACED = "oc_self_paced"
16+
EDX_COURSE = "oc_self_paced"
17+
18+
STAGE_PRODUCT_TYPE_ID_MAPPING = {
19+
ProductType.EDX_COURSE_ENTITLEMENT.value: "12e5510c-a4d6-4301-9caf-17053e57ff71",
20+
ProductType.EDX_PROGRAM.value: "79fb6abe-8373-4dec-a8d1-51242b1798b8",
21+
ProductType.OC_SELF_PACED.value: "9f8ec882-043a-4225-8811-00ac5acfd580"
22+
}
23+
24+
PROD_PRODUCT_TYPE_ID_MAPPING = {
25+
ProductType.EDX_COURSE_ENTITLEMENT.value: "9f1f189a-4d79-4eaa-9c6e-cfcb61aa779f",
26+
ProductType.EDX_PROGRAM.value: "c6a2d629-a50e-4d88-bd01-ab05a0617eae",
27+
ProductType.EDX_COURSE.value: "b241ac79-fee2-461d-b714-8f3c4a1c4c0e"
28+
}
29+
30+
class Command(CommercetoolsAPIClientCommand):
31+
help = "Fetch and verify course attributes from CommerceTools"
32+
33+
def handle(self, *args, **options):
34+
# Specify product type to fetch
35+
product_type = ProductType.EDX_COURSE_ENTITLEMENT
36+
37+
# Fetch products based on type
38+
products = self.fetch_products(product_type)
39+
40+
# Write data to CSV
41+
self.write_attributes_to_csv(products, product_type)
42+
43+
def fetch_products(self, product_type):
44+
limit = 500
45+
offset = 0
46+
products = []
47+
48+
product_type_id = STAGE_PRODUCT_TYPE_ID_MAPPING.get(product_type.value)
49+
50+
while True:
51+
products_result = self.ct_api_client.base_client.products.query(
52+
limit=limit,
53+
offset=offset,
54+
where=f"productType(id=\"{product_type_id}\")"
55+
)
56+
for product in products_result.results:
57+
attributes = self.extract_product_attributes(product, product_type.value)
58+
products.extend(attributes)
59+
60+
if products_result.offset + products_result.limit >= products_result.total:
61+
break
62+
offset += limit
63+
64+
return products
65+
66+
def extract_product_attributes(self, product, product_type):
67+
# Extract common product-level attributes
68+
common_attributes = {
69+
"product_type": product_type,
70+
"product_id": product.id,
71+
"product_key": product.key,
72+
"published_status": product.master_data.published,
73+
"name": product.master_data.current.name.get('en-US', ''),
74+
"slug": product.master_data.current.slug.get('en-US', ''),
75+
"description": (
76+
product.master_data.current.description.get('en-US', '')
77+
if product.master_data.current.description
78+
else ''
79+
),
80+
"date_created": product.created_at,
81+
"master_variant_key": product.master_data.current.master_variant.key,
82+
"master_variant_sku": product.master_data.current.master_variant.sku,
83+
"master_variant_image_url": (
84+
product.master_data.current.master_variant.images[0].url
85+
if product.master_data.current.master_variant.images
86+
else None
87+
),
88+
}
89+
90+
product_rows = [] # This will hold the product and variant rows
91+
92+
# Add the master variant attributes
93+
if len(product.master_data.current.variants) == 0:
94+
master_variant_attributes = {attr.name: attr.value for attr in
95+
product.master_data.current.master_variant.attributes}
96+
product_rows.append({**common_attributes, **master_variant_attributes})
97+
98+
# Add attributes for each variant and create a separate row, including variant_key and variant_sku
99+
for variant in product.master_data.current.variants:
100+
variant_attributes = {attr.name: attr.value for attr in variant.attributes}
101+
variant_row = {
102+
**common_attributes,
103+
"variant_key": variant.key, # Add variant_key
104+
"variant_sku": variant.sku, # Add variant_sku
105+
"variant_image_url": (
106+
variant.images[0].url
107+
if variant.images
108+
else None
109+
),
110+
**variant_attributes,
111+
}
112+
# Create a new row for each variant, combining common product data with variant-specific attributes
113+
product_rows.append(variant_row)
114+
115+
return product_rows
116+
117+
def write_attributes_to_csv(self, products, product_type):
118+
if not products:
119+
print(f"No products found for type {product_type}.")
120+
return
121+
122+
# Dynamically extract all unique keys across all product dictionaries
123+
keys = set()
124+
for product in products:
125+
keys.update(product.keys())
126+
127+
# Convert keys set back to a list and sort them if you want a consistent order
128+
keys = sorted(list(keys))
129+
130+
# Define CSV filename with product type and date
131+
filename = f"{product_type.value}_attributes_{datetime.now().strftime('%Y%m%d')}.csv"
132+
133+
# Write to CSV
134+
with open(filename, "w", newline="") as output_file:
135+
dict_writer = csv.DictWriter(output_file, fieldnames=keys)
136+
dict_writer.writeheader()
137+
dict_writer.writerows(products)
138+
139+
print(f"\n\n\n\n\n\n\nCSV file '{filename}' written successfully with {len(products)} records.")

0 commit comments

Comments
 (0)