initBasti / Amazon2PlentySync (public) (License: GPLv3) (since 2019-01-27) (hash sha1)
Transfer your data from you Amazon Flatfile spreadsheet over to the Plentymarkets system. How to is included in the readme
List of commits:
Subject Hash Author Date (UTC)
Refactor CategoryChooser 562e6657c6fef89d0584731e54325cec013268a7 Sebastian Fricke 2020-01-15 14:42:11
Add category_config location to the script config 8698e4a99d63b06fde5c39787fc7d6f7400b9f47 Sebastian Fricke 2020-01-15 14:29:47
Refactor findConfig 321ae9d7edd69e8be0755cf5ba82289944d06ca3 Sebastian Fricke 2020-01-15 14:26:09
Add logging function: no category config warning e8323843a3b6c24ef274d6a12c10d76aa8b8f591 Sebastian Fricke 2020-01-14 14:38:39
Add module + test for the category-id config fadaf4515aab1009f4df4a1af5a2e8f82077bc4c Sebastian Fricke 2020-01-14 14:35:44
improved coding style on log functions caf97eec6c6026aa051bc98f02a90e649a6e4754 Sebastian Fricke 2020-01-14 10:23:17
fixed a typo in the product type list 707e993b953aea0d653ffbca136bbe81bb36ea13 Sebastian Fricke 2020-01-14 10:22:34
added home product properties, improved dictionary iteration, fixed a key error in get_attributes 30d4aed4403c39a6865e30c0384c3360d375cbb6 Sebastian Fricke 2020-01-14 10:21:56
removed warning for missing flatfile columns as requested bfe6e22f7acb282a3af8423c81ceacf9fcf21ef4 Sebastian Fricke 2020-01-13 15:05:27
added initialization for the position variable 8331f92d62deb9ba7be7e97201c7c6afa7cf732a Sebastian Fricke 2020-01-13 14:47:57
improved code style and fixed problem where the dictionary containing the path was given to functions instead of the path itself 1a5bf99751b599f48d4687a9a6cbd55ffe213f5a Sebastian Fricke 2020-01-13 14:47:13
removed Barcode missing warning on parents b592779c6cc1588e2ae40394cab53d0d047746e7 Sebastian Fricke 2020-01-13 14:46:16
Added support for the amazon product types furnitureanddecor, bedandbath, swimwear b56708e55c3283a6cc2d3803b2abbb99bb125928 Sebastian Fricke 2020-01-13 14:16:40
fix failing attribute sync 87ea4bce17eba6c9c9842eaf9eb26249bf8d7da5 Sebastian Fricke 2020-01-13 12:15:35
new config handling d65bdfae89eceab6b1319d01373cf70ac7d8b63e Sebastian Fricke 2019-11-13 08:57:14
Fixed a problem, that caused Data to not pass sorting; Fixed error handling with the product type; Updated category ids 9a62d369fb24bc80765cd19e31fb255398fb8ed5 Sebastian Fricke 2019-09-12 09:27:54
fixed a merge conflict bug e6b4d9613237009d980cdbfc7ec65c3383a3495a Sebastian Fricke 2019-08-16 11:31:02
current status 15.8 94db3a5c98c596b24f00624fa4b772b9fd830b03 Sebastian Fricke 2019-08-15 14:26:42
Added manual file choosing in case of empty config 2df178528d70be15bfb2e1c9058f69e128236622 Sebastian Fricke 2019-08-15 10:11:41
Added Markdown choosing, fixed various bugs 991ed44df370cf80fc2e2c51d7427d63e221888f Sebastian Fricke 2019-08-15 09:30:55
Commit 562e6657c6fef89d0584731e54325cec013268a7 - Refactor CategoryChooser
Added the functionality to read the valid values for the dropdown from
the category config file, removed the activity category dropdown and
combine both dropdown lists to one, improve name style to use
camel case instead of split by '_'
Author: Sebastian Fricke
Author date (UTC): 2020-01-15 14:42
Committer name: Sebastian Fricke
Committer date (UTC): 2020-01-15 14:42
Parent(s): 8698e4a99d63b06fde5c39787fc7d6f7400b9f47
Signer:
Signing key:
Signing status: N
Tree: fa40db4daf92ae8e5bd09c922cc967a1b1a0f626
File Lines added Lines deleted
packages/gui/category_chooser.py 36 93
File packages/gui/category_chooser.py changed (mode: 100644) (index 3a7d7dc..a39e3c2)
... ... class MarkingDropdown(tkinter.Frame):
34 34 *[*self.options]) *[*self.options])
35 35 self.dropdown_menu.grid(row=1, column=0, sticky="EW", padx=50) self.dropdown_menu.grid(row=1, column=0, sticky="EW", padx=50)
36 36
37 self.optionvar.trace('w', self.change_dropdown)
37 self.optionvar.trace('w', self.changeDropdown)
38 38
39 def change_dropdown(self, *args):
39 def changeDropdown(self, *args):
40 40 if(self.optionvar.get() and not(self.optionvar.get() == 'marking')): if(self.optionvar.get() and not(self.optionvar.get() == 'marking')):
41 41 self.resultvar = self.options[self.optionvar.get()] self.resultvar = self.options[self.optionvar.get()]
42 42
 
... ... class InfoBox(tkinter.Frame):
108 108 self.dropdownbox = DropdownBox(self) self.dropdownbox = DropdownBox(self)
109 109 self.dropdownbox.grid(row=1, column=1, columnspan=2, sticky="EW") self.dropdownbox.grid(row=1, column=1, columnspan=2, sticky="EW")
110 110
111 self.optionvar.trace('w', self.change_dropdown)
111 self.optionvar.trace('w', self.changeDropdown)
112 112
113 113 self.labelbox = tkinter.Label(self, text="similar colors") self.labelbox = tkinter.Label(self, text="similar colors")
114 114 self.labelbox.grid(row=2, column=1, columnspan=3, sticky="NESW") self.labelbox.grid(row=2, column=1, columnspan=3, sticky="NESW")
115 115
116 def change_dropdown(self, *args):
116 def changeDropdown(self, *args):
117 117 if(self.labelbox.winfo_exists() == 1): if(self.labelbox.winfo_exists() == 1):
118 118 self.labelbox.destroy() self.labelbox.destroy()
119 119 for color in self.colordict: for color in self.colordict:
 
... ... class ButtonBox(tkinter.Frame):
135 135
136 136 self.continue_button = tkinter.Button(self, text="Continue", self.continue_button = tkinter.Button(self, text="Continue",
137 137 command=lambda: command=lambda:
138 self.master.destroy_warningbox())
138 self.master.destroyWarningbox())
139 139 self.continue_button.grid(row=1, column=1, sticky='EW', padx=20) self.continue_button.grid(row=1, column=1, sticky='EW', padx=20)
140 140
141 141 self.create_button = tkinter.Button(self, text="Create a Syncfile", self.create_button = tkinter.Button(self, text="Create a Syncfile",
142 142 command=lambda: command=lambda:
143 self.master.create_syncfile())
143 self.master.createSyncfile())
144 144 self.create_button.grid(row=1, column=2, sticky='EW', padx=20) self.create_button.grid(row=1, column=2, sticky='EW', padx=20)
145 145
146 146 self.stop_button = tkinter.Button(self, text='Stop', self.stop_button = tkinter.Button(self, text='Stop',
147 147 command=lambda: command=lambda:
148 self.master.destroy_root())
148 self.master.destroyRoot())
149 149 self.stop_button.grid(row=1, column=3, sticky='EW', padx=20) self.stop_button.grid(row=1, column=3, sticky='EW', padx=20)
150 150
151 151
 
... ... class WarningBox(tkinter.Frame):
167 167
168 168 if(re.search(r'list', str(type(self.colorlist)))): if(re.search(r'list', str(type(self.colorlist)))):
169 169 self.colorlist_length = len(self.colorlist) self.colorlist_length = len(self.colorlist)
170 self.colorlist_string = self.create_colorstring(self.colorlist)
170 self.colorlist_string = self.createColorstring(self.colorlist)
171 171 else: else:
172 172 print("ERROR @ WarningBox: colorlist needs to be a list\n{0}" print("ERROR @ WarningBox: colorlist needs to be a list\n{0}"
173 173 .format(type(self.colorlist))) .format(type(self.colorlist)))
 
... ... class WarningBox(tkinter.Frame):
195 195 self.infobox.grid(row=4, column=1, columnspan=3, self.infobox.grid(row=4, column=1, columnspan=3,
196 196 sticky='NESW', pady=20, padx=20) sticky='NESW', pady=20, padx=20)
197 197
198 def create_syncfile(self):
198 def createSyncfile(self):
199 199 clr.create_attributesync(self.colordict, self.master.master.upath) clr.create_attributesync(self.colordict, self.master.master.upath)
200 200 self.master.withdraw() self.master.withdraw()
201 201
202 def destroy_warningbox(self):
202 def destroyWarningbox(self):
203 203 self.master.withdraw() self.master.withdraw()
204 204
205 def destroy_root(self):
205 def destroyRoot(self):
206 206 self.master.master.destroy() self.master.master.destroy()
207 207
208 def create_colorstring(self, colorlist):
208 def createColorstring(self, colorlist):
209 209 colorstring = '' colorstring = ''
210 210
211 211 if(len(colorlist) > 3): if(len(colorlist) > 3):
 
... ... class DropdownChooser(tkinter.Frame):
231 231 self.activityvar = tkinter.StringVar(self) self.activityvar = tkinter.StringVar(self)
232 232 self.resultvar = tkinter.StringVar(self) self.resultvar = tkinter.StringVar(self)
233 233
234 self.options = {'Men.Aladinhose': '34',
235 'Men.Sommerhose': '36',
236 'Men.Stoffhose': '35',
237 'Men.Shorts': '37',
238 'Men.Fischerhose': '85',
239 'Men.Hemden': '38',
240 'Men.Tshirt': '39',
241 'Men.Hoodie': '40',
242 'Men.Jacken': '41',
243 'Women.Kleid': '62',
244 'Women.Tunika': '63',
245 'Women.Sarong-K': '64',
246 'Women.Skirt': '87',
247 'Women.Aladinhose': '53',
248 'Women.Stoffhose': '70',
249 'Women.Sommerhose': '71',
250 'Women.Hosenrock': '72',
251 'Women.Legging': '73',
252 'Women.Fischerhose': '96',
253 'Women.Top': '65',
254 'Women.Tshirt/Hemden': '66',
255 'Women.Hoodie': '68',
256 'Women.Sarong-O': '69',
257 'Women.Jacken': '84',
258 'Unisex.Bags': '108'
259 }
260
261 self.activities = {
262 'Women.Yoga': '95',
263 'Women.Retreat': '96',
264 'Women.Summerwear': '97',
265 'Women.Festival': '98',
266 'Men.Yoga': '99',
267 'Men.Retreat': '100',
268 'Men.Summerwear': '101',
269 'Men.Festival': '102'
270 }
234 self.options = self.master.cat
271 235
272 236 self.optionvar.set('category') self.optionvar.set('category')
273 self.activityvar.set('activities')
274 237
275 self.menu_header_major = tkinter.Label(self, text='Major category')
238 self.menu_header_major = tkinter.Label(self, text='category')
276 239 self.menu_header_major.grid(row=0, column=0, sticky='NESW') self.menu_header_major.grid(row=0, column=0, sticky='NESW')
277 240
278 self.menu_header_activity = tkinter.Label(self,
279 text='Activity category')
280 self.menu_header_activity.grid(row=0, column=1, sticky='NESW')
281
282 241 self.dropdown_menu = tkinter.OptionMenu(self, self.optionvar, self.dropdown_menu = tkinter.OptionMenu(self, self.optionvar,
283 242 *[*self.options]) *[*self.options])
284 243 self.dropdown_menu.grid(row=1, column=0, sticky="EW", padx=50) self.dropdown_menu.grid(row=1, column=0, sticky="EW", padx=50)
285 244
286 self.activity_menu = tkinter.OptionMenu(self, self.activityvar,
287 *[*self.activities])
288 self.activity_menu.grid(row=1, column=1, sticky="EW", padx=50)
289
290 self.optionvar.trace('w', self.change_dropdown)
291 self.activityvar.trace('w', self.change_dropdown)
292 self.resultvar.trace('w', self.add_desc)
245 self.optionvar.trace('w', self.changeDropdown)
246 self.resultvar.trace('w', self.addDesc)
293 247
294 248 # Create a textbox to show the result of the choosing # Create a textbox to show the result of the choosing
295 249 self.resultbox = tkinter.Entry(self, textvariable=self.resultvar, self.resultbox = tkinter.Entry(self, textvariable=self.resultvar,
 
... ... class DropdownChooser(tkinter.Frame):
303 257 tkinter.Label(self, tkinter.Label(self,
304 258 text="first entry used as standard category!") text="first entry used as standard category!")
305 259
306 def change_dropdown(self, *args):
260 def changeDropdown(self, *args):
307 261 if(not(self.resultbox.get())): if(not(self.resultbox.get())):
308 if(not(self.activityvar.get() == 'activities')):
309 print(self.activityvar.get())
310 tmb.showerror("No Major category!",
311 "Please enter first a major category, so that the first gets set as standard category. After that add a activity category!")
312 self.activityvar.set('activities')
313
314 262 self. resultbox.insert(tkinter.INSERT, self. resultbox.insert(tkinter.INSERT,
315 263 self.options[self.optionvar.get()]) self.options[self.optionvar.get()])
316 264 else: else:
317 265 if(self.optionvar.get() and if(self.optionvar.get() and
318 not(re.search(self.options[self.optionvar.get()],
319 self.resultbox.get()))):
266 not(self.options[self.optionvar.get()] in
267 [int(i) for i in self.resultbox.get().strip(' ').split(',')])):
320 268 self.resultbox.insert(tkinter.INSERT, ', ' + self.resultbox.insert(tkinter.INSERT, ', ' +
321 self.options[self.optionvar.get()])
322 if(self.activityvar.get() and
323 not(self.activityvar.get() == 'activities')):
324 if(not(re.search(self.activities[self.activityvar.get()],
325 self.resultbox.get()))):
326 self.resultbox.insert(tkinter.INSERT, ', ' +
327 self.activities[self.activityvar.
328 get()])
329
330 def add_desc(self, *args):
269 str(self.options[self.optionvar.get()]))
270
271 def addDesc(self, *args):
331 272 if(len(self.resultvar.get()) > 1): if(len(self.resultvar.get()) > 1):
332 273 self.category_info.grid(row=3, columnspan=2, sticky="EW") self.category_info.grid(row=3, columnspan=2, sticky="EW")
333 274 else: else:
 
... ... class DescBox(tkinter.Frame):
349 290
350 291
351 292 class CategoryChooser(tkinter.Tk): class CategoryChooser(tkinter.Tk):
352 def __init__(self, master, upath='', flatfile='', atrpath='', atrdate=''):
293 def __init__(self, master, cat, upath='', flatfile='',
294 atrpath='', atrdate=''):
353 295 tkinter.Tk.__init__(self, master) tkinter.Tk.__init__(self, master)
354 296 self.master = master self.master = master
355 297 self.upath = upath self.upath = upath
356 298 self.flatfile = flatfile self.flatfile = flatfile
357 299 self.atrpath = atrpath self.atrpath = atrpath
358 300 self.atrdate = atrdate self.atrdate = atrdate
301 self.cat = cat
359 302 self.newpath = {'upload-path': '', 'attribute-path': ''} self.newpath = {'upload-path': '', 'attribute-path': ''}
360 303 self.data = {'name': '', 'categories': '', 'marking': ''} self.data = {'name': '', 'categories': '', 'marking': ''}
361 self.protocol("WM_WINDOW_DELETE", self.close_app)
304 self.protocol("WM_WINDOW_DELETE", self.closeApp)
362 305 self.missingcolors = {} self.missingcolors = {}
363 306 # Window position properties # Window position properties
364 307 self.window_w = self.winfo_reqwidth() self.window_w = self.winfo_reqwidth()
 
... ... class CategoryChooser(tkinter.Tk):
383 326 self.grid() self.grid()
384 327
385 328 if(self.atrpath['path']): if(self.atrpath['path']):
386 self.check_colors(self.flatfile, self.atrpath)
329 self.checkColors(self.flatfile, self.atrpath)
387 330
388 331 self.pathdesc = DescBox(master=self, self.pathdesc = DescBox(master=self,
389 332 desctext="The current Upload path is: \n" + desctext="The current Upload path is: \n" +
 
... ... class CategoryChooser(tkinter.Tk):
399 342 self.changepathbutton =\ self.changepathbutton =\
400 343 tkinter.Button(self, text="Choose a new path", tkinter.Button(self, text="Choose a new path",
401 344 command=lambda: command=lambda:
402 self.get_new_path("Choose a new upload folder",
345 self.getNewPath("Choose a new upload folder",
403 346 'upload')) 'upload'))
404 347 self.changepathbutton.grid(row=2, column=1, pady=10, padx=10) self.changepathbutton.grid(row=2, column=1, pady=10, padx=10)
405 348
406 349 self.changeatrbutton =\ self.changeatrbutton =\
407 350 tkinter.Button(self, text="Choose the attribute file", tkinter.Button(self, text="Choose the attribute file",
408 351 command=lambda: command=lambda:
409 self.get_new_path("Choose a new attribute file",
352 self.getNewPath("Choose a new attribute file",
410 353 'atr')) 'atr'))
411 354 self.changeatrbutton.grid(row=2, column=2, pady=10, padx=10) self.changeatrbutton.grid(row=2, column=2, pady=10, padx=10)
412 355
 
... ... class CategoryChooser(tkinter.Tk):
435 378 self.accept =\ self.accept =\
436 379 tkinter.Button(self, text="Accept", tkinter.Button(self, text="Accept",
437 380 command=lambda: command=lambda:
438 self.get_input(self.dropdown.resultbox.get(),
381 self.getInput(self.dropdown.resultbox.get(),
439 382 self.namechooser.get(), self.namechooser.get(),
440 383 self.markingchooser.resultvar)) self.markingchooser.resultvar))
441 384 self.accept.grid(row=9, column=3, pady=10, padx=10) self.accept.grid(row=9, column=3, pady=10, padx=10)
442 385
443 def get_input(self, categories, name, marking):
386 def getInput(self, categories, name, marking):
444 387 self.data['name'] = name self.data['name'] = name
445 388 self.data['categories'] = categories self.data['categories'] = categories
446 389 self.data['marking'] = marking self.data['marking'] = marking
447 390 # Close the gui after accepting the input to stop the mainloop # Close the gui after accepting the input to stop the mainloop
448 self.close_app()
391 self.closeApp()
449 392
450 def get_new_path(self, title, option):
393 def getNewPath(self, title, option):
451 394 if(option == 'upload'): if(option == 'upload'):
452 395 print("Get the new path of the upload folder.") print("Get the new path of the upload folder.")
453 396 self.newpath['upload-path'] =\ self.newpath['upload-path'] =\
 
... ... class CategoryChooser(tkinter.Tk):
457 400 self.newpath['attribute-path'] =\ self.newpath['attribute-path'] =\
458 401 tkinter.filedialog.askopenfilename(title=title) tkinter.filedialog.askopenfilename(title=title)
459 402 self.atrdate = datetime.datetime.now().strftime("%d.%m.%Y-%H:%M") self.atrdate = datetime.datetime.now().strftime("%d.%m.%Y-%H:%M")
460 self.check_colors(self.flatfile, self.newpath['attribute-path'])
403 self.checkColors(self.flatfile, self.newpath['attribute-path'])
461 404
462 def check_colors(self, flatfile, attributefile):
405 def checkColors(self, flatfile, attributefile):
463 406 attributefile = item_upload.check_encoding(attributefile) attributefile = item_upload.check_encoding(attributefile)
464 407 self.missingcolors = clr.missingColor(flatfile, attributefile) self.missingcolors = clr.missingColor(flatfile, attributefile)
465 408
 
... ... class CategoryChooser(tkinter.Tk):
480 423 print("Error @ checkcolor: {0} - line no.: {1}" print("Error @ checkcolor: {0} - line no.: {1}"
481 424 .format(err, sys.exc_info()[2].tb_lineno)) .format(err, sys.exc_info()[2].tb_lineno))
482 425
483 def close_app(self):
426 def closeApp(self):
484 427 if(self.data['name'] and self.data['categories']): if(self.data['name'] and self.data['categories']):
485 428 self.withdraw() self.withdraw()
486 429 else: else:
Hints:
Before first commit, do not forget to setup your git environment:
git config --global user.name "your_name_here"
git config --global user.email "your@email_here"

Clone this repository using HTTP(S):
git clone https://rocketgit.com/user/initBasti/Amazon2PlentySync

Clone this repository using ssh (do not forget to upload a key first):
git clone ssh://rocketgit@ssh.rocketgit.com/user/initBasti/Amazon2PlentySync

Clone this repository using git:
git clone git://git.rocketgit.com/user/initBasti/Amazon2PlentySync

You are allowed to anonymously push to this repository.
This means that your pushed commits will automatically be transformed into a merge request:
... clone the repository ...
... make some changes and some commits ...
git push origin main