Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merchandise sales moneybird integration #3464

Closed
wants to merge 30 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
23246b3
Implement merchandise sale in backend and to mb
tmgdejong Oct 16, 2023
f74f5cb
parent cb8b4d9d631465e54739de5f58d58fa85eedf785
tmgdejong Oct 17, 2023
6ee7177
Correct ledger and small fixes
tmgdejong Oct 18, 2023
23f8672
Textual changes
tmgdejong Oct 19, 2023
c708f32
Remove TPay and Wire as options
tmgdejong Oct 18, 2023
26e69ca
Switch merchandise sales to sales
tmgdejong Oct 26, 2023
c4aa4c5
Continuing merch sales
tmgdejong Oct 26, 2023
393cbeb
Continuing merch sales
tmgdejong Oct 26, 2023
3859882
Small changes
tmgdejong Oct 30, 2023
9dc3e03
Merge branch 'master' into merchandise-sales-
tmgdejong Oct 30, 2023
97d2aa2
Fix migrations
tmgdejong Oct 30, 2023
f3e58f3
Removed comment
tmgdejong Oct 30, 2023
2cca5d7
Merge branch 'merchandise-sales-moneybird-integration' of github.com:…
tmgdejong Oct 30, 2023
c76f085
Some small changes wrt classes and names
tmgdejong Oct 31, 2023
9a8b60b
Small fix
tmgdejong Oct 31, 2023
4e32bea
Merge branch 'master' into merchandise-sales-moneybird-integration
tmgdejong Oct 31, 2023
4df93b9
Added attribute id
tmgdejong Nov 1, 2023
28f853f
correct api call for attr id
tmgdejong Nov 1, 2023
a95af92
Fixed migration
tmgdejong Nov 1, 2023
8f3fee0
Small fixes
tmgdejong Nov 1, 2023
0868f73
Add merchandise item syncing to Moneybird
tmgdejong Nov 6, 2023
ccb02cd
Merge branch 'master' into merchandise-sales-moneybird-integration
tmgdejong Nov 6, 2023
8a8b28f
Remove unused import
tmgdejong Nov 6, 2023
f75feb3
Rewrite to FKs
JobDoesburg Nov 7, 2023
540dc55
Move changes to merchandise
DeD1rk Nov 7, 2023
73fe1ce
Fix admin
JobDoesburg Nov 12, 2023
864f33a
Fix admin
JobDoesburg Nov 12, 2023
2ce0433
Fix moneybird
JobDoesburg Nov 12, 2023
67509f0
Fix bugs
JobDoesburg Nov 12, 2023
7b337ed
Sync individual items
JobDoesburg Nov 25, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
94 changes: 93 additions & 1 deletion website/merchandise/admin.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,26 @@
"""Registers admin interfaces for the models defined in this module."""

from django.contrib import admin
from django.contrib.admin import ModelAdmin
from django.shortcuts import redirect
from django.urls import reverse

from sales.admin import ProductAdmin, ProductListAdmin, ProductListItemInline
from sales.models.product import Product, ProductList

from .models import MerchandiseItem, MerchandiseProduct


from .models import MerchandiseItem
class MerchandiseProductInline(admin.TabularInline):
"""Inline admin interface for the merchandise products."""

model = MerchandiseProduct
extra = 0

fields = (
"name",
"stock_value",
)


@admin.register(MerchandiseItem)
Expand All @@ -16,3 +34,77 @@ class MerchandiseItemAdmin(ModelAdmin):
"description",
"image",
)
search_fields = ("name", "description")
list_display = ("name", "price")
list_filter = ("name", "price")

inlines = [MerchandiseProductInline]


class MerchandiseDisabledProductAdmin(ProductAdmin):
def has_change_permission(self, request, obj=None):
if obj is not None and isinstance(obj.merchandiseproduct, MerchandiseProduct):
return False
return super().has_change_permission(request, obj)

def has_delete_permission(self, request, obj=None):
if obj is not None and isinstance(obj.merchandiseproduct, MerchandiseProduct):
return False
return super().has_change_permission(request, obj)

def change_view(self, request, object_id, form_url="", extra_context=None):
obj = self.get_object(request, object_id)
if obj is not None and isinstance(obj.merchandiseproduct, MerchandiseProduct):
return redirect(
reverse(
"admin:merchandise_merchandiseitem_change",
args=(obj.merchandiseproduct.merchandise_item.id,),
)
)
return super().change_view(request, object_id, form_url, extra_context)


admin.site.unregister(Product)
admin.site.register(Product, MerchandiseDisabledProductAdmin)


class MerchandiseDisabledProductListItemInline(ProductListItemInline):
fields = (
"product",
"price",
"priority",
)

def has_delete_permission(self, request, obj=None):
return False

def has_add_permission(self, request, obj):
return False

def get_readonly_fields(self, request, obj=None):
readonly_fields = super().get_readonly_fields(request, obj)
return readonly_fields + (
"product",
"price",
)


class MerchandiseDisabledProductListAdmin(ProductListAdmin):
inlines = [
MerchandiseDisabledProductListItemInline,
]

def get_readonly_fields(self, request, obj=None):
readonly_fields = super().get_readonly_fields(request, obj)
if obj is not None and obj.name == "Merchandise":
return readonly_fields + ("name",)
return readonly_fields

def has_delete_permission(self, request, obj=None):
if obj is not None and obj.name == "Merchandise":
return False
return super().has_change_permission(request, obj)


admin.site.unregister(ProductList)
admin.site.register(ProductList, MerchandiseDisabledProductListAdmin)
3 changes: 3 additions & 0 deletions website/merchandise/apps.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,9 @@ class MerchandiseConfig(AppConfig):
name = "merchandise"
verbose_name = _("Merchandise")

def ready(self):
from . import signals # noqa: F401

def menu_items(self):
return {
"categories": [{"name": "association", "title": "Association", "key": 1}],
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
# Generated by Django 4.2.6 on 2023-11-12 21:51

from django.db import migrations, models
import django.db.models.deletion
import payments.models


class Migration(migrations.Migration):
dependencies = [
("sales", "0007_alter_productlistitem_options_and_more"),
("merchandise", "0011_alter_merchandiseitem_image"),
]

operations = [
migrations.AlterField(
model_name="merchandiseitem",
name="price",
field=payments.models.PaymentAmountField(
decimal_places=2,
help_text="Current sales price of the merchandise item per piece (incl. VAT).",
max_digits=8,
validators=[payments.models.validate_not_zero],
),
),
migrations.CreateModel(
name="MerchandiseProduct",
fields=[
(
"product_ptr",
models.OneToOneField(
auto_created=True,
on_delete=django.db.models.deletion.CASCADE,
parent_link=True,
primary_key=True,
serialize=False,
to="sales.product",
),
),
(
"stock_value",
payments.models.PaymentAmountField(
decimal_places=2,
help_text="Current stock ledger value of this product per piece (excl. VAT).",
max_digits=8,
validators=[payments.models.validate_not_zero],
),
),
(
"merchandise_item",
models.ForeignKey(
on_delete=django.db.models.deletion.CASCADE,
to="merchandise.merchandiseitem",
),
),
],
bases=("sales.product",),
),
]
25 changes: 21 additions & 4 deletions website/merchandise/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@

from thumbnails.fields import ImageField

from payments.models import PaymentAmountField
from sales.models.product import Product
from utils.media.services import get_upload_to_function

_merchandise_photo_upload_to = get_upload_to_function("merchandise")
Expand All @@ -15,13 +17,11 @@ class MerchandiseItem(models.Model):
This model describes merchandise items.
"""

#: Name of the merchandise item.
name = models.CharField(max_length=200)

#: Price of the merchandise item
price = models.DecimalField(
max_digits=8,
decimal_places=2,
price = PaymentAmountField(
help_text="Current sales price of the merchandise item per piece (incl. VAT)."
)

#: Description of the merchandise item
Expand Down Expand Up @@ -61,3 +61,20 @@ def __str__(self):
:rtype: str
"""
return str(self.name)


class MerchandiseProduct(Product):
"""Merchandise products."""

merchandise_item = models.ForeignKey(
MerchandiseItem,
on_delete=models.CASCADE,
)

stock_value = PaymentAmountField(
help_text="Current stock ledger value of this product per piece (excl. VAT)."
)

def __str__(self):
"""Give the name of the merchandise product in the currently active locale."""
return f"{self.name} ({self.merchandise_item})"
48 changes: 48 additions & 0 deletions website/merchandise/services.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import datetime

from django.utils import timezone

from activemembers.models import Board
from merchandise.models import MerchandiseProduct
from sales.models.product import ProductList
from sales.models.shift import Shift


def update_merchandise_product_list():
product_list = ProductList.objects.get_or_create(name="Merchandise")[0]
merchandise_products = MerchandiseProduct.objects.all()

for merchandise_product in merchandise_products:
item, _ = product_list.product_items.get_or_create(
product=merchandise_product,
defaults={"price": merchandise_product.merchandise_item.price},
)
item.price = merchandise_product.merchandise_item.price
item.save()

return product_list


def create_daily_merchandise_sale_shift():
today = timezone.now().date()
merchandise_product_list = update_merchandise_product_list()
active_board = Board.objects.filter(since__lte=today, until__gte=today)

shift = Shift.objects.create(
title="Merchandise sales",
start=timezone.now(),
end=timezone.datetime.combine(today, datetime.time(23, 59, 59)),
product_list=merchandise_product_list,
)
shift.managers.set(active_board)
shift.save()


def lock_merchandise_sale_shift():
shifts = Shift.objects.filter(title="Merchandise sales").all()
for shift in shifts:
if shift.num_orders == 0:
shift.delete()
else:
shift.locked = True
shift.save()
12 changes: 12 additions & 0 deletions website/merchandise/signals.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
from django.db.models.signals import post_save
from django.dispatch import receiver

from merchandise.models import MerchandiseItem, MerchandiseProduct

from .services import update_merchandise_product_list


@receiver(post_save, sender=MerchandiseProduct)
@receiver(post_save, sender=MerchandiseItem)
def update_merchandise_product_list_on_save(sender, instance, **kwargs):
update_merchandise_product_list()
9 changes: 9 additions & 0 deletions website/merchandise/tasks.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
from celery import shared_task

from . import services


@shared_task
def renew_merchandise_sale_shift():
services.lock_merchandise_sale_shift()
services.create_daily_merchandise_sale_shift()
39 changes: 38 additions & 1 deletion website/moneybirdsynchronization/admin.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,12 @@
from django.contrib import admin
from django.contrib.admin import RelatedOnlyFieldListFilter

from .models import MoneybirdContact, MoneybirdExternalInvoice, MoneybirdPayment
from .models import (
MoneybirdContact,
MoneybirdExternalInvoice,
MoneybirdMerchandiseSaleJournal,
MoneybirdPayment,
)


@admin.register(MoneybirdContact)
Expand Down Expand Up @@ -100,3 +105,35 @@ def get_readonly_fields(self, request, obj=None):
if not obj:
return ()
return ("payment",)


@admin.register(MoneybirdMerchandiseSaleJournal)
class MoneybirdMerchandiseSaleJournalAdmin(admin.ModelAdmin):
list_display = (
"order",
"moneybird_general_journal_document_id",
"needs_synchronization",
"needs_deletion",
)

readonly_fields = (
"order",
"moneybird_general_journal_document_id",
"moneybird_details_debit_attribute_id",
"moneybird_details_credit_attribute_id",
"needs_synchronization",
"needs_deletion",
)

fields = (
"order",
"moneybird_general_journal_document_id",
"moneybird_details_debit_attribute_id",
"moneybird_details_credit_attribute_id",
"external_invoice",
)

list_filter = (
"needs_synchronization",
"needs_deletion",
)
Loading