from outils.common_imports import * from outils.for_view_imports import * from .models import Shelf, Shelfs from inventory.models import CagetteInventory from datetime import date from datetime import datetime def index(request): """Main shelf page""" shelfs = Shelfs.get_all() # TODO : Make the distinction beetween active and inactive products for s in shelfs: s['shelf_value'] = -1 context = {'title': 'Rayons', 'shelfs': json.dumps(shelfs)} template = loader.get_template('shelfs/index.html') return HttpResponse(template.render(context, request)) def sales(request): """Main sales view""" shelfs = Shelfs.get_all() context = {'title': 'Rayons (Ventes)', 'shelfs': json.dumps(shelfs), 'SHELFS_SCRIPT': 'shelfs_sales'} template = loader.get_template('shelfs/index.html') return HttpResponse(template.render(context, request)) def shelf_view(request, id): """Page for shelf inventory""" shelf_products = Shelf(id).get_products() context = {'title': 'Vue du rayon', 'shelf_products': json.dumps(shelf_products['data'])} template = loader.get_template('shelfs/shelf_view.html') return HttpResponse(template.render(context, request)) def shelf_inventory(request, id): """Page for shelf inventory""" shelf_products = Shelf(id).get_products() context = {'title': 'Inventaire du rayon', 'products': json.dumps(shelf_products['data']), 'ahead_shelfs_ids': json.dumps(getattr(settings, 'SHELFS_TO_BE_AHEAD_IN_SELECT_LIST', [])) } template = loader.get_template('shelfs/shelf_inventory.html') return HttpResponse(template.render(context, request)) def shelf_data(request, shelf_id): """Get a shelf data""" shelf = Shelf(shelf_id).get() if 'error' in shelf: return JsonResponse(shelf, status=500) else: return JsonResponse({'res': shelf}) def set_begin_inventory_datetime(request, shelf_id): """ Set the ongoing inventory start datetime. Set it to now. """ res = Shelf(shelf_id).set_begin_inventory_datetime() if 'error' in res: return JsonResponse(res, status=500) else: return JsonResponse({'res': res}) def delete_ongoing_inv_data(request, shelf_id): m = Shelf(shelf_id) res = m.delete_ongoing_inv_data() if 'error' in res: return JsonResponse(res, status=500) else: return JsonResponse({'res': res}) def all(request, precision): """Get all shelves data""" return JsonResponse({'res': Shelfs.get_all(precision)}) def get_shelves_extra_data(request): """Get data that need calculation, so long execution time""" shelfs = Shelfs.get_all() res = [] for s in shelfs : shelf_products = Shelf(s['id']).get_products()['data'] shelf_value = 0 for p in shelf_products : shelf_value += float(p['qty_available']) * float(p['standard_price']) res.append({ 'id': s['id'], 'shelf_value': round(shelf_value, 2) }) return JsonResponse({'res': res}) def products(request, shelf_id): """Get all products from a shelf""" res = {} try: res = Shelf(shelf_id).get_products() except Exception as e: res['error'] = str(e) return JsonResponse({'res': res}) def add_product(request, shelf_id): """Add a product to a shelf""" res = {} params = json.loads(request.body.decode()) barcodes = [params['barcode']] res = Shelf(shelf_id).add_products_by_barcodes(barcodes) if 'error' in res or 'msg' in res: return JsonResponse({'res': res}, status=500) return JsonResponse({'res': res}) def inventory_process_state(request, shelf_id): res = {} try: s = Shelf(shelf_id) s_data = s.get() res['state'] = s_data['inventory_status'] except Exception as e: res['error'] = str(e) coop_logger.error("Inventory process state : %s", str(e)) if 'error' in res: return JsonResponse(res, status=500) else: return JsonResponse({'res': res}) def change_products_shelfs(request): res = {} try: data = json.loads(request.body.decode()) res = Shelfs.make_products_shelf_links(data) except Exception as e: res['error'] = str(e) coop_logger.error("change_products_shelfs : %s", str(e)) if 'error' in res: return JsonResponse(res, status=500) else: return JsonResponse({'res': res}) def do_shelf_inventory(request): """Process shelf inventory""" """ If many products are implied, the whole process could last many minutes. During this time, user can submit data again. This is managed with 'busy' message returned by Shelf.get_full_inventory_data method Web server can also return a timeout message during this time. This is managed by sending a query to above "get_process_state" (within browser ajax error capture) """ res = {} # TODO : manage error strings array instead of one string try: shelf_data = json.loads(request.body.decode()) m = Shelf(shelf_data['id']) # Set inventory data inventory_date = date.today() inventory_data = { 'name': shelf_data['name'] + ' - ' + inventory_date.strftime("%d/%m/%Y"), 'shelf_id': shelf_data['id'], 'user_comments': shelf_data['user_comments'], 'products': shelf_data['list_processed'], 'status': shelf_data['inventory_status'] } try: filename = 'data/inventories_backup/' filename += datetime.today().strftime("%Y-%m-%d--%H-%M-%S") filename += "__" + str(shelf_data['id']) + '.json' with open(filename, 'w') as outfile: json.dump(shelf_data, outfile) except Exception as serr: coop_logger.error("Inventory backup failure : %s", str(serr)) try: if shelf_data['inventory_status'] == '': # First step: save first products count in temp file res = m.save_tmp_inventory(inventory_data) else: inventory_data['date'] = inventory_date inventory_data['shelf_name'] = shelf_data['name'] inventory_data['shelf_num'] = shelf_data['sort_order'] # Get data from step 1 full_inventory_data = m.get_full_inventory_data(inventory_data) if 'error' in full_inventory_data: res['error'] = full_inventory_data['error'] if 'busy' in full_inventory_data: res['busy'] = True return JsonResponse(res, status=500) # Proceed with inventory res['inventory'] = CagetteInventory.update_products_stock(full_inventory_data) full_inventory_data['inventory_id'] = res['inventory']['inv_id'] shelf_data['last_inventory_id'] = res['inventory']['inv_id'] # Save inventory report res['inv_report'] = m.save_inventory_report(full_inventory_data) shelf_data['shelf_delta'] = res['inv_report']['shelf_delta'] shelf_data['shelf_losses'] = res['inv_report']['shelf_losses'] # Remove temp file from step 1 m.remove_tmp_inventory() try: # Update products with inventory data res['products_data_update'] = m.update_products_with_inventory_data(res['inv_report']['products']) except Exception as ee: res['error'] = {'products_data_update': str(ee)} coop_logger.error("products_data_update : %s, %s", res['inv_report']['products'], str(ee)) try: # Update shelf inventory data res['shelf'] = m.update_shelf_with_inventory_data(shelf_data) except Exception as ee: res['error'] = {'shelf': str(ee)} coop_logger.error("Shelf inv. data : %s, %s", shelf_data, str(ee)) except Exception as e: # Don't validate if error anywhere in inventory process res['error'] = str(e) coop_logger.error("Shelf inv. : %s", str(e)) except Exception as err_json: res['error'] = "Unable to parse received JSON" received = "?" try: received = '"' + request.body.decode() + '"' except Exception as err_rb: received = str(err_rb) coop_logger.error("Unable to parse received JSON : %s, %s", received , err_json) if 'error' in res: return JsonResponse(res, status=500) else: return JsonResponse({'res': res}) def get_last_inventory_report(request, shelf_id): import base64, os res = {} try: m_sh = Shelf(shelf_id) shelf = m_sh.get() if shelf['last_inventory_id'] != 0: m_inv = CagetteInventory(shelf['last_inventory_id']) file = m_inv.get_file() response = HttpResponse(base64.b64decode(file['datas']), content_type=file['mimetype']) response['Content-Disposition'] = 'attachment; filename="' + file['name'] + '"' return response else: res['error'] = 'Ce rayon n\'a pas encore été inventorié' return JsonResponse(res, status=500) except Exception as e: res['error'] = str(e) return JsonResponse(res, status=500) def shelf_inventory_FAQ(request): """Send content of the FAQ""" context = {} template = loader.get_template('shelfs/shelf_inventory_FAQ.html') return HttpResponse(template.render(context, request))