txn_view.py 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. #
  2. # Copyright (c) Daniel Sheffield 2021 - 2022
  3. #
  4. # All rights reserved
  5. #
  6. # THIS SOFTWARE IS PROVIDED AS IS WITHOUT WARRANTY
  7. from psycopg.sql import (
  8. Identifier,
  9. SQL,
  10. Literal,
  11. Placeholder,
  12. Composed,
  13. )
  14. from collections import (
  15. OrderedDict,
  16. )
  17. def get_table_statement(alias):
  18. tables = {
  19. 'product': 'products',
  20. 'category': 'categories',
  21. 'group': 'groups',
  22. 'unit': 'units',
  23. 'store': 'stores',
  24. }
  25. return SQL("SELECT {column} AS {alias} FROM {table}").format(
  26. column=Identifier(tables[alias], 'name'),
  27. table=Identifier(tables[alias]),
  28. alias=Identifier(alias),
  29. )
  30. def get_transactions_statement():
  31. return SQL("SELECT * FROM transaction_view")
  32. select = OrderedDict([
  33. ('id', Identifier('transactions','id')),
  34. ('ts', Identifier('transactions', 'ts')),
  35. ('store', Identifier('stores', 'name')),
  36. ('code', Identifier('stores', 'code')),
  37. ('description', Identifier('transactions','description')),
  38. ('volume', Identifier('quantity')),
  39. ('unit', Identifier('units', 'name')),
  40. ('price', Identifier('price')),
  41. ('$/unit', SQL("""TRUNC(price/quantity,4)""")),
  42. ('total', SQL("""sum(transactions.price) OVER (PARTITION BY transactions.ts::date ORDER BY transactions.id ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW)""")),
  43. ('group', Identifier('groups','name')),
  44. ('category', Identifier('categories','name')),
  45. ('product', Identifier('products','name')),
  46. ('organic', Identifier('organic')),
  47. ])
  48. def get_where(date, store, full_name=False, exact_time=False):
  49. where = [ ]
  50. if store is not None:
  51. where.append(SQL(' ').join([
  52. Identifier('stores', 'name' if full_name else 'code'),
  53. SQL('='),
  54. Literal(store)
  55. ]))
  56. where.append(
  57. SQL("{ts} at time zone 'utc' BETWEEN {date}::date AND {date}::date + {interval}::interval").format(
  58. ts=Identifier('ts'),
  59. date=Literal(str(date)),
  60. interval=Literal('23 hours 59 minutes 59 seconds'),
  61. ) if not exact_time else SQL("{ts} at time zone 'utc' = {date}").format(
  62. ts=Identifier('ts'),
  63. date=Literal(str(date)),
  64. )
  65. )
  66. return SQL('').join([
  67. SQL("WHERE"
  68. "\n "),
  69. SQL("\n AND ").join(where),
  70. ])
  71. def get_session_transactions_statement(date, store, full_name=False, exact_time=False):
  72. statement = SQL('\n').join([
  73. SQL('').join([
  74. SQL("SELECT"
  75. "\n "),
  76. SQL(','
  77. "\n ").join([
  78. SQL(' ').join([v, SQL('AS'), Identifier(f'{k}')]) for k, v in select.items()
  79. ])
  80. ]),
  81. SQL('').join([
  82. SQL("FROM"
  83. "\n "),
  84. SQL("\n JOIN ").join([
  85. SQL("transactions"),
  86. SQL("{table} ON {table}.{key} = {index}").format(
  87. table=Identifier('units'),
  88. key=Identifier('id'),
  89. index=Identifier('unit_id'),
  90. ),
  91. SQL("{table} ON {table}.{key} = {index}").format(
  92. table=Identifier('stores'),
  93. key=Identifier('id'),
  94. index=Identifier('store_id')
  95. ),
  96. SQL("{table} ON {table}.{key} = {index}").format(
  97. table=Identifier('products'),
  98. key=Identifier('id'),
  99. index=Identifier('product_id')
  100. ),
  101. SQL("{table} ON {table}.{key} = {index}").format(
  102. table=Identifier('categories'),
  103. key=Identifier('id'),
  104. index=Identifier('category_id')
  105. ),
  106. SQL("{table} ON {table}.{key} = {index}").format(
  107. table=Identifier('groups'),
  108. key=Identifier('id'),
  109. index=Identifier('group_id')
  110. ),
  111. ]),
  112. ]),
  113. get_where(date if exact_time else date.replace(
  114. hour=0, minute=0, second=0, microsecond=0
  115. ), store, full_name=full_name, exact_time=exact_time),
  116. SQL('ORDER BY {_id} DESC').format(_id=Identifier('transactions','id')),
  117. ])
  118. return statement