packages/ flattens into: backends/ server protocol adapters (mizan-django, mizan-ts) frontends/ client kernel + framework adapters (mizan-base, mizan-react, mizan-vue, mizan-svelte) workers/ runtime workers (mizan-ssr) cores/ shared language-level primitives (empty for now; mizan-python forthcoming) The frontend kernel (was packages/mizan-runtime, now frontends/mizan-base) is renamed to reflect its role — it's the shared base that frontend adapters depend on directly. Reflects the substrate position that per-framework adapters wrap a single shared kernel; codegen targets the adapter, not the raw kernel. Path updates landed in: Makefile, two Gitea workflows, Dockerfile.test, four example/harness config files, .claude/settings.local.json, four docs (CLAUDE/ISSUES/ROADMAP/AFI_ARCHITECTURE), four codegen templates (stage1 + react/vue/svelte adapters), and three package.jsons (the mizan-base rename plus mizan-vue/svelte peerDeps). Generated files under examples/django-react-site/harness/src/api/ still reference @mizan/runtime — left as-is; they're regenerated artifacts and the harness is non-functional pending the React wrapper-layer codegen. Also folded in a pre-existing fix: the Gitea workflows had working-directory: react / django pointing at a layout that predates packages/, never updated. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
143 lines
4.1 KiB
Python
143 lines
4.1 KiB
Python
from django.contrib.auth.models import (
|
|
AbstractBaseUser,
|
|
BaseUserManager,
|
|
PermissionsMixin,
|
|
)
|
|
from django.db import models
|
|
|
|
|
|
class EmailUserManager(BaseUserManager):
|
|
"""Custom user manager using email as the unique identifier."""
|
|
|
|
def create_user(self, email, password=None, **extra_fields):
|
|
if not email:
|
|
raise ValueError("Email is required")
|
|
email = self.normalize_email(email)
|
|
user = self.model(email=email, **extra_fields)
|
|
user.set_password(password)
|
|
user.save(using=self._db)
|
|
return user
|
|
|
|
def create_superuser(self, email, password=None, **extra_fields):
|
|
extra_fields.setdefault("is_staff", True)
|
|
extra_fields.setdefault("is_superuser", True)
|
|
return self.create_user(email, password, **extra_fields)
|
|
|
|
|
|
class EmailUser(AbstractBaseUser, PermissionsMixin):
|
|
"""Minimal user model with email as USERNAME_FIELD.
|
|
|
|
Matches the calling convention used in mizan's test suite:
|
|
User.objects.create_user(email="...", password="...", is_staff=True)
|
|
"""
|
|
|
|
email = models.EmailField(unique=True)
|
|
is_staff = models.BooleanField(default=False)
|
|
is_active = models.BooleanField(default=True)
|
|
|
|
objects = EmailUserManager()
|
|
|
|
USERNAME_FIELD = "email"
|
|
REQUIRED_FIELDS = []
|
|
|
|
class Meta:
|
|
app_label = "tests"
|
|
|
|
|
|
# ─── Shape test models ──────────────────────────────────────────────────────
|
|
|
|
import uuid
|
|
|
|
|
|
class TimestampMixin(models.Model):
|
|
created_at = models.DateTimeField(auto_now_add=True)
|
|
updated_at = models.DateTimeField(auto_now=True)
|
|
|
|
class Meta:
|
|
abstract = True
|
|
|
|
|
|
class Publisher(TimestampMixin):
|
|
name = models.CharField(max_length=200)
|
|
country = models.CharField(max_length=100, default="")
|
|
|
|
class Meta:
|
|
app_label = "tests"
|
|
|
|
|
|
class Author(TimestampMixin):
|
|
name = models.CharField(max_length=200)
|
|
bio = models.TextField(default="")
|
|
publisher = models.ForeignKey(
|
|
Publisher, on_delete=models.CASCADE, related_name="authors"
|
|
)
|
|
mentor = models.ForeignKey(
|
|
"self", on_delete=models.SET_NULL, null=True, blank=True, related_name="mentees"
|
|
)
|
|
|
|
class Meta:
|
|
app_label = "tests"
|
|
|
|
|
|
class Tag(models.Model):
|
|
slug = models.SlugField(primary_key=True, max_length=100)
|
|
label = models.CharField(max_length=100)
|
|
|
|
class Meta:
|
|
app_label = "tests"
|
|
|
|
|
|
class Book(TimestampMixin):
|
|
title = models.CharField(max_length=300)
|
|
isbn = models.CharField(max_length=13, unique=True)
|
|
page_count = models.IntegerField(default=0)
|
|
is_published = models.BooleanField(default=False)
|
|
author = models.ForeignKey(Author, on_delete=models.CASCADE, related_name="books")
|
|
editor = models.ForeignKey(
|
|
Author,
|
|
on_delete=models.SET_NULL,
|
|
null=True,
|
|
blank=True,
|
|
related_name="edited_books",
|
|
)
|
|
tags = models.ManyToManyField(Tag, blank=True, related_name="books")
|
|
|
|
class Meta:
|
|
app_label = "tests"
|
|
|
|
|
|
class Chapter(TimestampMixin):
|
|
book = models.ForeignKey(Book, on_delete=models.CASCADE, related_name="chapters")
|
|
number = models.IntegerField()
|
|
title = models.CharField(max_length=300)
|
|
word_count = models.IntegerField(default=0)
|
|
|
|
class Meta:
|
|
app_label = "tests"
|
|
ordering = ["number"]
|
|
unique_together = [("book", "number")]
|
|
|
|
|
|
class Section(models.Model):
|
|
id = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False)
|
|
chapter = models.ForeignKey(
|
|
Chapter, on_delete=models.CASCADE, related_name="sections"
|
|
)
|
|
heading = models.CharField(max_length=300)
|
|
body = models.TextField(default="")
|
|
position = models.IntegerField(default=0)
|
|
|
|
class Meta:
|
|
app_label = "tests"
|
|
ordering = ["position"]
|
|
|
|
|
|
class Category(models.Model):
|
|
name = models.CharField(max_length=200)
|
|
parent = models.ForeignKey(
|
|
"self", on_delete=models.CASCADE, null=True, blank=True, related_name="children"
|
|
)
|
|
|
|
class Meta:
|
|
app_label = "tests"
|