1063 lines
43 KiB
Python
1063 lines
43 KiB
Python
"""
|
|
Admin interface for auth models
|
|
"""
|
|
from django import forms
|
|
from django.contrib import admin
|
|
from django.contrib.auth.admin import UserAdmin as BaseUserAdmin
|
|
from unfold.admin import ModelAdmin, TabularInline
|
|
from simple_history.admin import SimpleHistoryAdmin
|
|
from igny8_core.admin.base import AccountAdminMixin, Igny8ModelAdmin
|
|
from .models import User, Account, Plan, Subscription, Site, Sector, SiteUserAccess, Industry, IndustrySector, SeedKeyword, PasswordResetToken
|
|
from import_export.admin import ExportMixin, ImportExportMixin
|
|
from import_export import resources
|
|
|
|
|
|
class AccountAdminForm(forms.ModelForm):
|
|
"""Custom form for Account admin with dynamic payment method choices from PaymentMethodConfig"""
|
|
|
|
class Meta:
|
|
model = Account
|
|
fields = '__all__'
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
|
|
from igny8_core.business.billing.models import PaymentMethodConfig, AccountPaymentMethod
|
|
|
|
if self.instance and self.instance.pk:
|
|
# Get country from billing_country, fallback to wildcard '*' for global
|
|
country = self.instance.billing_country or '*'
|
|
|
|
# Get enabled payment methods for this country OR global (*)
|
|
available_methods = PaymentMethodConfig.objects.filter(
|
|
country_code__in=[country, '*'],
|
|
is_enabled=True
|
|
).order_by('country_code', 'sort_order').values_list('payment_method', 'display_name')
|
|
|
|
if available_methods:
|
|
# Build choices from PaymentMethodConfig
|
|
choices = []
|
|
seen = set()
|
|
for method_type, display_name in available_methods:
|
|
if method_type not in seen:
|
|
choices.append((method_type, display_name or method_type.replace('_', ' ').title()))
|
|
seen.add(method_type)
|
|
else:
|
|
# Fallback to model choices if no configs
|
|
choices = Account.PAYMENT_METHOD_CHOICES
|
|
|
|
self.fields['payment_method'].widget = forms.Select(choices=choices)
|
|
|
|
# Get current default from AccountPaymentMethod
|
|
default_method = AccountPaymentMethod.objects.filter(
|
|
account=self.instance,
|
|
is_default=True,
|
|
is_enabled=True
|
|
).first()
|
|
|
|
if default_method:
|
|
self.fields['payment_method'].initial = default_method.type
|
|
self.fields['payment_method'].help_text = f'✓ Current: {default_method.display_name} ({default_method.get_type_display()})'
|
|
else:
|
|
self.fields['payment_method'].help_text = 'Select from available payment methods based on country'
|
|
|
|
def save(self, commit=True):
|
|
"""When payment_method changes, update/create AccountPaymentMethod"""
|
|
from igny8_core.business.billing.models import AccountPaymentMethod, PaymentMethodConfig
|
|
|
|
instance = super().save(commit=False)
|
|
|
|
if commit:
|
|
instance.save()
|
|
|
|
# Get selected payment method
|
|
selected_type = self.cleaned_data.get('payment_method')
|
|
|
|
if selected_type:
|
|
# Get config for display name and instructions
|
|
country = instance.billing_country or '*'
|
|
config = PaymentMethodConfig.objects.filter(
|
|
country_code__in=[country, '*'],
|
|
payment_method=selected_type,
|
|
is_enabled=True
|
|
).first()
|
|
|
|
# Create or update AccountPaymentMethod
|
|
account_method, created = AccountPaymentMethod.objects.get_or_create(
|
|
account=instance,
|
|
type=selected_type,
|
|
defaults={
|
|
'display_name': config.display_name if config else selected_type.replace('_', ' ').title(),
|
|
'is_default': True,
|
|
'is_enabled': True,
|
|
'instructions': config.instructions if config else '',
|
|
'country_code': instance.billing_country or '',
|
|
}
|
|
)
|
|
|
|
if not created:
|
|
# Update existing and set as default
|
|
account_method.is_default = True
|
|
account_method.is_enabled = True
|
|
if config:
|
|
account_method.display_name = config.display_name
|
|
account_method.instructions = config.instructions
|
|
account_method.save()
|
|
|
|
# Unset other methods as default
|
|
AccountPaymentMethod.objects.filter(
|
|
account=instance
|
|
).exclude(id=account_method.id).update(is_default=False)
|
|
|
|
return instance
|
|
|
|
|
|
class PlanResource(resources.ModelResource):
|
|
"""Resource class for importing/exporting Plans"""
|
|
class Meta:
|
|
model = Plan
|
|
fields = ('id', 'name', 'slug', 'price', 'billing_cycle', 'max_sites', 'max_users',
|
|
'max_keywords', 'max_ahrefs_queries', 'included_credits', 'is_active', 'is_featured')
|
|
export_order = fields
|
|
import_id_fields = ('id',)
|
|
skip_unchanged = True
|
|
|
|
|
|
@admin.register(Plan)
|
|
class PlanAdmin(ImportExportMixin, Igny8ModelAdmin):
|
|
resource_class = PlanResource
|
|
"""Plan admin - Global, no account filtering needed"""
|
|
list_display = ['name', 'slug', 'price', 'billing_cycle', 'max_sites', 'max_users', 'max_keywords', 'max_ahrefs_queries', 'included_credits', 'is_active', 'is_featured']
|
|
list_filter = ['is_active', 'billing_cycle', 'is_internal', 'is_featured']
|
|
search_fields = ['name', 'slug']
|
|
readonly_fields = ['created_at']
|
|
actions = [
|
|
'bulk_set_active',
|
|
'bulk_set_inactive',
|
|
'bulk_clone_plans',
|
|
]
|
|
|
|
fieldsets = (
|
|
('Plan Info', {
|
|
'fields': ('name', 'slug', 'price', 'original_price', 'annual_discount_percent', 'billing_cycle', 'features', 'is_active', 'is_featured', 'is_internal'),
|
|
'description': 'Price: Current price | Original Price: Crossed-out price (optional) | Annual Discount %: For annual billing | Is Featured: Show as popular/recommended plan'
|
|
}),
|
|
('Account Management Limits', {
|
|
'fields': ('max_users', 'max_sites', 'max_industries', 'max_author_profiles'),
|
|
'description': 'Persistent limits for account-level resources'
|
|
}),
|
|
('Hard Limits (Persistent)', {
|
|
'fields': ('max_keywords',),
|
|
'description': 'Total allowed - never reset'
|
|
}),
|
|
('Monthly Limits (Reset on Billing Cycle)', {
|
|
'fields': ('max_ahrefs_queries',),
|
|
'description': 'Monthly Ahrefs keyword research queries (0 = disabled)'
|
|
}),
|
|
('Billing & Credits', {
|
|
'fields': ('included_credits', 'extra_credit_price', 'allow_credit_topup', 'auto_credit_topup_threshold', 'auto_credit_topup_amount', 'credits_per_month')
|
|
}),
|
|
('Stripe Integration', {
|
|
'fields': ('stripe_product_id', 'stripe_price_id')
|
|
}),
|
|
)
|
|
|
|
def bulk_set_active(self, request, queryset):
|
|
"""Set selected plans to active"""
|
|
updated = queryset.update(is_active=True)
|
|
self.message_user(request, f'{updated} plan(s) set to active.', messages.SUCCESS)
|
|
bulk_set_active.short_description = 'Set plans to Active'
|
|
|
|
def bulk_set_inactive(self, request, queryset):
|
|
"""Set selected plans to inactive"""
|
|
updated = queryset.update(is_active=False)
|
|
self.message_user(request, f'{updated} plan(s) set to inactive.', messages.SUCCESS)
|
|
bulk_set_inactive.short_description = 'Set plans to Inactive'
|
|
|
|
def bulk_clone_plans(self, request, queryset):
|
|
"""Clone selected plans"""
|
|
count = 0
|
|
for plan in queryset:
|
|
plan_copy = Plan.objects.get(pk=plan.pk)
|
|
plan_copy.pk = None
|
|
plan_copy.name = f"{plan.name} (Copy)"
|
|
plan_copy.slug = f"{plan.slug}-copy"
|
|
plan_copy.is_active = False
|
|
plan_copy.save()
|
|
count += 1
|
|
self.message_user(request, f'{count} plan(s) cloned.', messages.SUCCESS)
|
|
bulk_clone_plans.short_description = 'Clone selected plans'
|
|
|
|
|
|
class AccountResource(resources.ModelResource):
|
|
"""Resource class for exporting Accounts"""
|
|
class Meta:
|
|
model = Account
|
|
fields = ('id', 'name', 'slug', 'owner__email', 'plan__name', 'status',
|
|
'credits', 'billing_country', 'created_at', 'updated_at')
|
|
export_order = fields
|
|
|
|
|
|
@admin.register(Account)
|
|
class AccountAdmin(ExportMixin, AccountAdminMixin, SimpleHistoryAdmin, Igny8ModelAdmin):
|
|
resource_class = AccountResource
|
|
form = AccountAdminForm
|
|
list_display = ['name', 'slug', 'owner', 'plan', 'status', 'health_indicator', 'credits', 'created_at']
|
|
list_filter = ['status', 'plan']
|
|
search_fields = ['name', 'slug']
|
|
readonly_fields = ['created_at', 'updated_at', 'health_indicator', 'health_details']
|
|
actions = [
|
|
'bulk_set_status_active',
|
|
'bulk_set_status_suspended',
|
|
'bulk_set_status_trial',
|
|
'bulk_set_status_cancelled',
|
|
'bulk_add_credits',
|
|
'bulk_subtract_credits',
|
|
'bulk_soft_delete',
|
|
]
|
|
|
|
def get_queryset(self, request):
|
|
"""Override to filter by account for non-superusers"""
|
|
qs = super().get_queryset(request)
|
|
if request.user.is_superuser or (hasattr(request.user, 'is_developer') and request.user.is_developer()):
|
|
return qs
|
|
# Owners can see their own accounts
|
|
if hasattr(request.user, 'role') and request.user.role == 'owner':
|
|
return qs.filter(owner=request.user)
|
|
# Admins can see their account
|
|
try:
|
|
user_account = getattr(request.user, 'account', None)
|
|
if user_account:
|
|
return qs.filter(id=user_account.id)
|
|
except (AttributeError, Exception):
|
|
# If account access fails (e.g., column mismatch), return empty
|
|
pass
|
|
return qs.none()
|
|
|
|
def health_indicator(self, obj):
|
|
"""Display health status with visual indicator"""
|
|
from django.utils.html import format_html
|
|
from django.utils import timezone
|
|
from datetime import timedelta
|
|
|
|
# Check credits
|
|
if obj.credits < 10:
|
|
status = 'critical'
|
|
message = 'Critical: Very low credits'
|
|
elif obj.credits < 100:
|
|
status = 'warning'
|
|
message = 'Warning: Low credits'
|
|
else:
|
|
status = 'good'
|
|
message = 'Good'
|
|
|
|
# Check for recent failed automations
|
|
try:
|
|
from igny8_core.business.automation.models import AutomationRun
|
|
week_ago = timezone.now() - timedelta(days=7)
|
|
failed_runs = AutomationRun.objects.filter(
|
|
account=obj,
|
|
status='failed',
|
|
created_at__gte=week_ago
|
|
).count()
|
|
|
|
if failed_runs > 5:
|
|
status = 'critical'
|
|
message = f'Critical: {failed_runs} automation failures'
|
|
elif failed_runs > 0:
|
|
if status == 'good':
|
|
status = 'warning'
|
|
message = f'Warning: {failed_runs} automation failures'
|
|
except:
|
|
pass
|
|
|
|
# Check account status
|
|
if obj.status != 'active':
|
|
status = 'critical'
|
|
message = f'Critical: Account {obj.status}'
|
|
|
|
colors = {
|
|
'good': '#0bbf87',
|
|
'warning': '#ff7a00',
|
|
'critical': '#ef4444'
|
|
}
|
|
|
|
return format_html(
|
|
'<span style="display: inline-flex; align-items: center; padding: 4px 12px; border-radius: 6px; background-color: {}; color: white; font-weight: 500; font-size: 13px;">{}</span>',
|
|
colors[status], message
|
|
)
|
|
health_indicator.short_description = 'Health'
|
|
|
|
def health_details(self, obj):
|
|
"""Detailed health information"""
|
|
from django.utils.html import format_html
|
|
from django.utils import timezone
|
|
from datetime import timedelta
|
|
|
|
details = []
|
|
|
|
# Credits status
|
|
colors = {
|
|
'critical': '#ef4444',
|
|
'warning': '#ff7a00',
|
|
'good': '#0bbf87'
|
|
}
|
|
|
|
if obj.credits < 10:
|
|
details.append(f'<span style="color: {colors["critical"]}; font-weight: 600;">Critical:</span> Only {obj.credits} credits remaining')
|
|
elif obj.credits < 100:
|
|
details.append(f'<span style="color: {colors["warning"]}; font-weight: 600;">Warning:</span> Only {obj.credits} credits remaining')
|
|
else:
|
|
details.append(f'<span style="color: {colors["good"]}; font-weight: 600;">Credits:</span> {obj.credits} available')
|
|
|
|
# Recent activity
|
|
try:
|
|
from igny8_core.modules.writer.models import Content
|
|
week_ago = timezone.now() - timedelta(days=7)
|
|
recent_content = Content.objects.filter(
|
|
site__account=obj,
|
|
created_at__gte=week_ago
|
|
).count()
|
|
details.append(f'📚 <b>Activity:</b> {recent_content} content pieces created this week')
|
|
except:
|
|
pass
|
|
|
|
# Failed automations
|
|
try:
|
|
from igny8_core.business.automation.models import AutomationRun
|
|
week_ago = timezone.now() - timedelta(days=7)
|
|
failed_runs = AutomationRun.objects.filter(
|
|
account=obj,
|
|
status='failed',
|
|
created_at__gte=week_ago
|
|
).count()
|
|
|
|
if failed_runs > 0:
|
|
details.append(f'🔴 <b>Automations:</b> {failed_runs} failures this week')
|
|
else:
|
|
details.append(f'✅ <b>Automations:</b> No failures this week')
|
|
except:
|
|
pass
|
|
|
|
# Failed syncs
|
|
try:
|
|
from igny8_core.business.integration.models import SyncEvent
|
|
today = timezone.now().date()
|
|
failed_syncs = SyncEvent.objects.filter(
|
|
site__account=obj,
|
|
success=False,
|
|
created_at__date=today
|
|
).count()
|
|
|
|
if failed_syncs > 0:
|
|
details.append(f'⚠️ <b>Syncs:</b> {failed_syncs} failures today')
|
|
else:
|
|
details.append(f'✅ <b>Syncs:</b> No failures today')
|
|
except:
|
|
pass
|
|
|
|
# Account status
|
|
if obj.status == 'active':
|
|
details.append(f'✅ <b>Status:</b> Active')
|
|
else:
|
|
details.append(f'🔴 <b>Status:</b> {obj.status.title()}')
|
|
|
|
return format_html('<br>'.join(details))
|
|
health_details.short_description = 'Health Details'
|
|
|
|
def has_delete_permission(self, request, obj=None):
|
|
if obj and getattr(obj, 'slug', '') == 'aws-admin':
|
|
return False
|
|
return super().has_delete_permission(request, obj)
|
|
|
|
# Bulk Actions
|
|
def bulk_set_status_active(self, request, queryset):
|
|
"""Set selected accounts to active status"""
|
|
updated = queryset.update(status='active')
|
|
self.message_user(request, f'{updated} account(s) set to active.', messages.SUCCESS)
|
|
bulk_set_status_active.short_description = 'Set status to Active'
|
|
|
|
def bulk_set_status_suspended(self, request, queryset):
|
|
"""Set selected accounts to suspended status"""
|
|
updated = queryset.update(status='suspended')
|
|
self.message_user(request, f'{updated} account(s) set to suspended.', messages.SUCCESS)
|
|
bulk_set_status_suspended.short_description = 'Set status to Suspended'
|
|
|
|
def bulk_set_status_trial(self, request, queryset):
|
|
"""Set selected accounts to trial status"""
|
|
updated = queryset.update(status='trial')
|
|
self.message_user(request, f'{updated} account(s) set to trial.', messages.SUCCESS)
|
|
bulk_set_status_trial.short_description = 'Set status to Trial'
|
|
|
|
def bulk_set_status_cancelled(self, request, queryset):
|
|
"""Set selected accounts to cancelled status"""
|
|
updated = queryset.update(status='cancelled')
|
|
self.message_user(request, f'{updated} account(s) set to cancelled.', messages.SUCCESS)
|
|
bulk_set_status_cancelled.short_description = 'Set status to Cancelled'
|
|
|
|
def bulk_add_credits(self, request, queryset):
|
|
"""Add credits to selected accounts"""
|
|
from django import forms
|
|
|
|
if 'apply' in request.POST:
|
|
amount = int(request.POST.get('credits', 0))
|
|
if amount > 0:
|
|
for account in queryset:
|
|
account.credits += amount
|
|
account.save()
|
|
self.message_user(request, f'Added {amount} credits to {queryset.count()} account(s).', messages.SUCCESS)
|
|
return
|
|
|
|
class CreditForm(forms.Form):
|
|
credits = forms.IntegerField(
|
|
min_value=1,
|
|
label="Credits to Add",
|
|
help_text=f"Add credits to {queryset.count()} selected account(s)"
|
|
)
|
|
|
|
from django.shortcuts import render
|
|
return render(request, 'admin/bulk_action_form.html', {
|
|
'title': 'Add Credits to Accounts',
|
|
'queryset': queryset,
|
|
'form': CreditForm(),
|
|
'action': 'bulk_add_credits',
|
|
})
|
|
bulk_add_credits.short_description = 'Add credits to accounts'
|
|
|
|
def bulk_subtract_credits(self, request, queryset):
|
|
"""Subtract credits from selected accounts"""
|
|
from django import forms
|
|
|
|
if 'apply' in request.POST:
|
|
amount = int(request.POST.get('credits', 0))
|
|
if amount > 0:
|
|
for account in queryset:
|
|
account.credits = max(0, account.credits - amount)
|
|
account.save()
|
|
self.message_user(request, f'Subtracted {amount} credits from {queryset.count()} account(s).', messages.SUCCESS)
|
|
return
|
|
|
|
class CreditForm(forms.Form):
|
|
credits = forms.IntegerField(
|
|
min_value=1,
|
|
label="Credits to Subtract",
|
|
help_text=f"Subtract credits from {queryset.count()} selected account(s)"
|
|
)
|
|
|
|
from django.shortcuts import render
|
|
return render(request, 'admin/bulk_action_form.html', {
|
|
'title': 'Subtract Credits from Accounts',
|
|
'queryset': queryset,
|
|
'form': CreditForm(),
|
|
'action': 'bulk_subtract_credits',
|
|
})
|
|
bulk_subtract_credits.short_description = 'Subtract credits from accounts'
|
|
|
|
def bulk_soft_delete(self, request, queryset):
|
|
"""Soft delete selected accounts"""
|
|
count = 0
|
|
for account in queryset:
|
|
if account.slug != 'aws-admin': # Protect admin account
|
|
account.delete() # Soft delete via SoftDeletableModel
|
|
count += 1
|
|
self.message_user(request, f'{count} account(s) soft deleted.', messages.SUCCESS)
|
|
bulk_soft_delete.short_description = 'Soft delete selected accounts'
|
|
|
|
|
|
class SubscriptionResource(resources.ModelResource):
|
|
"""Resource class for exporting Subscriptions"""
|
|
class Meta:
|
|
model = Subscription
|
|
fields = ('id', 'account__name', 'status', 'current_period_start', 'current_period_end',
|
|
'stripe_subscription_id', 'created_at')
|
|
export_order = fields
|
|
|
|
|
|
@admin.register(Subscription)
|
|
class SubscriptionAdmin(ExportMixin, AccountAdminMixin, Igny8ModelAdmin):
|
|
resource_class = SubscriptionResource
|
|
list_display = ['account', 'status', 'current_period_start', 'current_period_end']
|
|
list_filter = ['status']
|
|
search_fields = ['account__name', 'stripe_subscription_id']
|
|
readonly_fields = ['created_at', 'updated_at']
|
|
actions = [
|
|
'bulk_set_status_active',
|
|
'bulk_set_status_cancelled',
|
|
'bulk_set_status_suspended',
|
|
'bulk_set_status_trialing',
|
|
'bulk_renew',
|
|
]
|
|
actions = [
|
|
'bulk_set_status_active',
|
|
'bulk_set_status_cancelled',
|
|
'bulk_set_status_suspended',
|
|
'bulk_set_status_trialing',
|
|
'bulk_renew',
|
|
]
|
|
|
|
def bulk_set_status_active(self, request, queryset):
|
|
"""Set subscriptions to active"""
|
|
updated = queryset.update(status='active')
|
|
self.message_user(request, f'{updated} subscription(s) set to active.', messages.SUCCESS)
|
|
bulk_set_status_active.short_description = 'Set status to Active'
|
|
|
|
def bulk_set_status_cancelled(self, request, queryset):
|
|
"""Set subscriptions to cancelled"""
|
|
updated = queryset.update(status='cancelled')
|
|
self.message_user(request, f'{updated} subscription(s) set to cancelled.', messages.SUCCESS)
|
|
bulk_set_status_cancelled.short_description = 'Set status to Cancelled'
|
|
|
|
def bulk_set_status_suspended(self, request, queryset):
|
|
"""Set subscriptions to suspended"""
|
|
updated = queryset.update(status='suspended')
|
|
self.message_user(request, f'{updated} subscription(s) set to suspended.', messages.SUCCESS)
|
|
bulk_set_status_suspended.short_description = 'Set status to Suspended'
|
|
|
|
def bulk_set_status_trialing(self, request, queryset):
|
|
"""Set subscriptions to trialing"""
|
|
updated = queryset.update(status='trialing')
|
|
self.message_user(request, f'{updated} subscription(s) set to trialing.', messages.SUCCESS)
|
|
bulk_set_status_trialing.short_description = 'Set status to Trialing'
|
|
|
|
def bulk_renew(self, request, queryset):
|
|
"""Renew selected subscriptions"""
|
|
from django.utils import timezone
|
|
from datetime import timedelta
|
|
|
|
count = 0
|
|
for subscription in queryset:
|
|
# Extend subscription by one billing period
|
|
if subscription.current_period_end:
|
|
subscription.current_period_end = subscription.current_period_end + timedelta(days=30)
|
|
subscription.status = 'active'
|
|
subscription.save()
|
|
count += 1
|
|
self.message_user(request, f'{count} subscription(s) renewed for 30 days.', messages.SUCCESS)
|
|
bulk_renew.short_description = 'Renew subscriptions'
|
|
|
|
|
|
@admin.register(PasswordResetToken)
|
|
class PasswordResetTokenAdmin(Igny8ModelAdmin):
|
|
list_display = ['user', 'token', 'used', 'expires_at', 'created_at']
|
|
list_filter = ['used', 'expires_at', 'created_at']
|
|
search_fields = ['user__email', 'token']
|
|
readonly_fields = ['created_at', 'token']
|
|
|
|
def get_queryset(self, request):
|
|
"""Filter by account for non-superusers"""
|
|
qs = super().get_queryset(request)
|
|
if request.user.is_superuser or (hasattr(request.user, 'is_developer') and request.user.is_developer()):
|
|
return qs
|
|
user_account = getattr(request.user, 'account', None)
|
|
if user_account:
|
|
return qs.filter(user__account=user_account)
|
|
return qs.none()
|
|
|
|
|
|
class SectorInline(TabularInline):
|
|
"""Inline admin for sectors within Site admin."""
|
|
model = Sector
|
|
extra = 0
|
|
fields = ['industry_sector', 'name', 'slug', 'status', 'is_active', 'get_keywords_count', 'get_clusters_count']
|
|
readonly_fields = ['get_keywords_count', 'get_clusters_count']
|
|
|
|
def get_keywords_count(self, obj):
|
|
if obj.pk:
|
|
try:
|
|
return obj.keywords_set.count()
|
|
except (AttributeError, Exception):
|
|
return 0
|
|
return 0
|
|
get_keywords_count.short_description = 'Keywords'
|
|
|
|
def get_clusters_count(self, obj):
|
|
if obj.pk:
|
|
try:
|
|
return obj.clusters_set.count()
|
|
except (AttributeError, Exception):
|
|
return 0
|
|
return 0
|
|
get_clusters_count.short_description = 'Clusters'
|
|
|
|
|
|
class SiteResource(resources.ModelResource):
|
|
"""Resource class for importing/exporting Sites"""
|
|
class Meta:
|
|
model = Site
|
|
fields = ('id', 'name', 'slug', 'account__name', 'industry__name', 'domain',
|
|
'status', 'is_active', 'site_type', 'hosting_type', 'description', 'created_at')
|
|
export_order = fields
|
|
import_id_fields = ('id',)
|
|
skip_unchanged = True
|
|
|
|
|
|
@admin.register(Site)
|
|
class SiteAdmin(ImportExportMixin, AccountAdminMixin, Igny8ModelAdmin):
|
|
resource_class = SiteResource
|
|
list_display = ['name', 'slug', 'account', 'industry', 'domain', 'status', 'is_active', 'get_api_key_status', 'get_sectors_count']
|
|
list_filter = ['status', 'is_active', 'account', 'industry', 'hosting_type']
|
|
search_fields = ['name', 'slug', 'domain', 'industry__name']
|
|
readonly_fields = ['created_at', 'updated_at', 'get_api_key_display']
|
|
inlines = [SectorInline]
|
|
actions = [
|
|
'generate_api_keys',
|
|
'bulk_set_status_active',
|
|
'bulk_set_status_inactive',
|
|
'bulk_set_status_maintenance',
|
|
'bulk_soft_delete',
|
|
]
|
|
|
|
fieldsets = (
|
|
('Site Info', {
|
|
'fields': ('name', 'slug', 'account', 'domain', 'description', 'industry', 'site_type', 'hosting_type', 'status', 'is_active')
|
|
}),
|
|
('WordPress Integration', {
|
|
'fields': ('get_api_key_display',),
|
|
'description': 'WordPress integration API key. Use SiteIntegration model for full integration settings.'
|
|
}),
|
|
('SEO Metadata', {
|
|
'fields': ('seo_metadata',),
|
|
'classes': ('collapse',)
|
|
}),
|
|
('Timestamps', {
|
|
'fields': ('created_at', 'updated_at'),
|
|
'classes': ('collapse',)
|
|
}),
|
|
)
|
|
|
|
def get_api_key_display(self, obj):
|
|
"""Display API key with copy button"""
|
|
from django.utils.html import format_html
|
|
if obj.wp_api_key:
|
|
return format_html(
|
|
'<div style="display:flex; align-items:center; gap:10px;">'
|
|
'<code style="background:#f0f0f0; padding:5px 10px; border-radius:3px;">{}</code>'
|
|
'<button type="button" onclick="navigator.clipboard.writeText(\'{}\'); alert(\'API Key copied to clipboard!\');" '
|
|
'style="padding:5px 10px; cursor:pointer;">Copy</button>'
|
|
'</div>',
|
|
obj.wp_api_key,
|
|
obj.wp_api_key
|
|
)
|
|
return format_html('<em>No API key generated</em>')
|
|
get_api_key_display.short_description = 'WordPress API Key'
|
|
|
|
def get_api_key_status(self, obj):
|
|
"""Show API key status in list view"""
|
|
from django.utils.html import format_html
|
|
if obj.wp_api_key:
|
|
return format_html('<span style="color:green;">●</span> Active')
|
|
return format_html('<span style="color:gray;">○</span> None')
|
|
get_api_key_status.short_description = 'API Key'
|
|
|
|
def generate_api_keys(self, request, queryset):
|
|
"""Generate API keys for selected sites"""
|
|
import secrets
|
|
updated_count = 0
|
|
for site in queryset:
|
|
if not site.wp_api_key:
|
|
site.wp_api_key = f"igny8_{''.join(secrets.choice('abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789') for _ in range(40))}"
|
|
site.save()
|
|
updated_count += 1
|
|
self.message_user(request, f'Generated API keys for {updated_count} site(s). Sites with existing keys were skipped.')
|
|
generate_api_keys.short_description = 'Generate WordPress API Keys'
|
|
|
|
def bulk_set_status_active(self, request, queryset):
|
|
"""Set selected sites to active status"""
|
|
updated = queryset.update(status='active', is_active=True)
|
|
self.message_user(request, f'{updated} site(s) set to active.', messages.SUCCESS)
|
|
bulk_set_status_active.short_description = 'Set status to Active'
|
|
|
|
def bulk_set_status_inactive(self, request, queryset):
|
|
"""Set selected sites to inactive status"""
|
|
updated = queryset.update(status='inactive', is_active=False)
|
|
self.message_user(request, f'{updated} site(s) set to inactive.', messages.SUCCESS)
|
|
bulk_set_status_inactive.short_description = 'Set status to Inactive'
|
|
|
|
def bulk_set_status_maintenance(self, request, queryset):
|
|
"""Set selected sites to maintenance status"""
|
|
updated = queryset.update(status='maintenance')
|
|
self.message_user(request, f'{updated} site(s) set to maintenance mode.', messages.SUCCESS)
|
|
bulk_set_status_maintenance.short_description = 'Set status to Maintenance'
|
|
|
|
def bulk_soft_delete(self, request, queryset):
|
|
"""Soft delete selected sites"""
|
|
count = 0
|
|
for site in queryset:
|
|
site.delete() # Soft delete via SoftDeletableModel
|
|
count += 1
|
|
self.message_user(request, f'{count} site(s) soft deleted.', messages.SUCCESS)
|
|
bulk_soft_delete.short_description = 'Soft delete selected sites'
|
|
|
|
def get_sectors_count(self, obj):
|
|
try:
|
|
return obj.get_active_sectors_count()
|
|
except:
|
|
return 0
|
|
get_sectors_count.short_description = 'Active Sectors'
|
|
|
|
def get_industry_display(self, obj):
|
|
"""Safely get industry name"""
|
|
try:
|
|
return obj.industry.name if obj.industry else '-'
|
|
except:
|
|
return '-'
|
|
get_industry_display.short_description = 'Industry'
|
|
|
|
|
|
class SectorResource(resources.ModelResource):
|
|
"""Resource class for exporting Sectors"""
|
|
class Meta:
|
|
model = Sector
|
|
fields = ('id', 'name', 'slug', 'site__name', 'industry_sector__name', 'status',
|
|
'is_active', 'created_at')
|
|
export_order = fields
|
|
|
|
|
|
@admin.register(Sector)
|
|
class SectorAdmin(ExportMixin, AccountAdminMixin, Igny8ModelAdmin):
|
|
resource_class = SectorResource
|
|
list_display = ['name', 'slug', 'site', 'industry_sector', 'get_industry', 'status', 'is_active', 'get_keywords_count', 'get_clusters_count']
|
|
list_filter = ['status', 'is_active', 'site', 'industry_sector__industry']
|
|
search_fields = ['name', 'slug', 'site__name', 'industry_sector__name']
|
|
readonly_fields = ['created_at', 'updated_at']
|
|
actions = [
|
|
'bulk_set_status_active',
|
|
'bulk_set_status_inactive',
|
|
'bulk_soft_delete',
|
|
]
|
|
|
|
def get_industry(self, obj):
|
|
"""Safely get industry name"""
|
|
try:
|
|
if obj.industry_sector and obj.industry_sector.industry:
|
|
return obj.industry_sector.industry.name
|
|
except:
|
|
pass
|
|
return '-'
|
|
get_industry.short_description = 'Industry'
|
|
|
|
def get_keywords_count(self, obj):
|
|
"""Safely get keywords count"""
|
|
try:
|
|
if obj.pk:
|
|
return getattr(obj, 'keywords_set', obj.keywords_set).count()
|
|
except:
|
|
pass
|
|
return 0
|
|
get_keywords_count.short_description = 'Keywords'
|
|
|
|
def get_clusters_count(self, obj):
|
|
"""Safely get clusters count"""
|
|
try:
|
|
if obj.pk:
|
|
return getattr(obj, 'clusters_set', obj.clusters_set).count()
|
|
except:
|
|
pass
|
|
return 0
|
|
get_clusters_count.short_description = 'Clusters'
|
|
|
|
def bulk_set_status_active(self, request, queryset):
|
|
"""Set selected sectors to active status"""
|
|
updated = queryset.update(status='active', is_active=True)
|
|
self.message_user(request, f'{updated} sector(s) set to active.', messages.SUCCESS)
|
|
bulk_set_status_active.short_description = 'Set status to Active'
|
|
|
|
def bulk_set_status_inactive(self, request, queryset):
|
|
"""Set selected sectors to inactive status"""
|
|
updated = queryset.update(status='inactive', is_active=False)
|
|
self.message_user(request, f'{updated} sector(s) set to inactive.', messages.SUCCESS)
|
|
bulk_set_status_inactive.short_description = 'Set status to Inactive'
|
|
|
|
def bulk_soft_delete(self, request, queryset):
|
|
"""Soft delete selected sectors"""
|
|
count = 0
|
|
for sector in queryset:
|
|
sector.delete() # Soft delete via SoftDeletableModel
|
|
count += 1
|
|
self.message_user(request, f'{count} sector(s) soft deleted.', messages.SUCCESS)
|
|
bulk_soft_delete.short_description = 'Soft delete selected sectors'
|
|
|
|
|
|
@admin.register(SiteUserAccess)
|
|
class SiteUserAccessAdmin(Igny8ModelAdmin):
|
|
list_display = ['user', 'site', 'granted_at', 'granted_by']
|
|
list_filter = ['granted_at']
|
|
search_fields = ['user__email', 'site__name']
|
|
readonly_fields = ['granted_at']
|
|
|
|
|
|
class IndustrySectorInline(TabularInline):
|
|
"""Inline admin for industry sectors within Industry admin."""
|
|
model = IndustrySector
|
|
extra = 0
|
|
fields = ['name', 'slug', 'description', 'is_active']
|
|
readonly_fields = []
|
|
|
|
|
|
class IndustryResource(resources.ModelResource):
|
|
"""Resource class for importing/exporting Industries"""
|
|
class Meta:
|
|
model = Industry
|
|
fields = ('id', 'name', 'slug', 'description', 'is_active', 'created_at')
|
|
export_order = fields
|
|
import_id_fields = ('id',)
|
|
skip_unchanged = True
|
|
|
|
|
|
@admin.register(Industry)
|
|
class IndustryAdmin(ImportExportMixin, Igny8ModelAdmin):
|
|
resource_class = IndustryResource
|
|
list_display = ['name', 'slug', 'is_active', 'get_sectors_count', 'created_at']
|
|
list_filter = ['is_active']
|
|
search_fields = ['name', 'slug', 'description']
|
|
readonly_fields = ['created_at', 'updated_at']
|
|
inlines = [IndustrySectorInline]
|
|
actions = [
|
|
'delete_selected',
|
|
'bulk_activate',
|
|
'bulk_deactivate',
|
|
] # Enable bulk delete
|
|
|
|
def get_sectors_count(self, obj):
|
|
return obj.sectors.filter(is_active=True).count()
|
|
get_sectors_count.short_description = 'Active Sectors'
|
|
|
|
def has_delete_permission(self, request, obj=None):
|
|
"""Allow deletion for superusers and developers"""
|
|
return request.user.is_superuser or (hasattr(request.user, 'is_developer') and request.user.is_developer())
|
|
|
|
def bulk_activate(self, request, queryset):
|
|
updated = queryset.update(is_active=True)
|
|
self.message_user(request, f'{updated} industry/industries activated.', messages.SUCCESS)
|
|
bulk_activate.short_description = 'Activate selected industries'
|
|
|
|
def bulk_deactivate(self, request, queryset):
|
|
updated = queryset.update(is_active=False)
|
|
self.message_user(request, f'{updated} industry/industries deactivated.', messages.SUCCESS)
|
|
bulk_deactivate.short_description = 'Deactivate selected industries'
|
|
|
|
|
|
class IndustrySectorResource(resources.ModelResource):
|
|
"""Resource class for importing/exporting Industry Sectors"""
|
|
class Meta:
|
|
model = IndustrySector
|
|
fields = ('id', 'name', 'slug', 'industry__name', 'description', 'is_active', 'created_at')
|
|
export_order = fields
|
|
import_id_fields = ('id',)
|
|
skip_unchanged = True
|
|
|
|
|
|
@admin.register(IndustrySector)
|
|
class IndustrySectorAdmin(ImportExportMixin, Igny8ModelAdmin):
|
|
resource_class = IndustrySectorResource
|
|
list_display = ['name', 'slug', 'industry', 'is_active']
|
|
list_filter = ['is_active', 'industry']
|
|
search_fields = ['name', 'slug', 'description']
|
|
readonly_fields = ['created_at', 'updated_at']
|
|
actions = [
|
|
'delete_selected',
|
|
'bulk_activate',
|
|
'bulk_deactivate',
|
|
] # Enable bulk delete
|
|
|
|
def has_delete_permission(self, request, obj=None):
|
|
"""Allow deletion for superusers and developers"""
|
|
return request.user.is_superuser or (hasattr(request.user, 'is_developer') and request.user.is_developer())
|
|
|
|
def bulk_activate(self, request, queryset):
|
|
updated = queryset.update(is_active=True)
|
|
self.message_user(request, f'{updated} sector(s) activated.', messages.SUCCESS)
|
|
bulk_activate.short_description = 'Activate selected sectors'
|
|
|
|
def bulk_deactivate(self, request, queryset):
|
|
updated = queryset.update(is_active=False)
|
|
self.message_user(request, f'{updated} sector(s) deactivated.', messages.SUCCESS)
|
|
bulk_deactivate.short_description = 'Deactivate selected sectors'
|
|
|
|
|
|
class SeedKeywordResource(resources.ModelResource):
|
|
"""Resource class for importing/exporting Seed Keywords"""
|
|
class Meta:
|
|
model = SeedKeyword
|
|
fields = ('id', 'keyword', 'industry__name', 'sector__name', 'volume',
|
|
'difficulty', 'country', 'is_active', 'created_at')
|
|
export_order = fields
|
|
import_id_fields = ('id',)
|
|
skip_unchanged = True
|
|
|
|
|
|
@admin.register(SeedKeyword)
|
|
class SeedKeywordAdmin(ImportExportMixin, Igny8ModelAdmin):
|
|
resource_class = SeedKeywordResource
|
|
"""SeedKeyword admin - Global reference data, no account filtering"""
|
|
list_display = ['keyword', 'industry', 'sector', 'volume', 'difficulty', 'country', 'is_active', 'created_at']
|
|
list_filter = ['is_active', 'industry', 'sector', 'country']
|
|
search_fields = ['keyword']
|
|
readonly_fields = ['created_at', 'updated_at']
|
|
actions = [
|
|
'delete_selected',
|
|
'bulk_activate',
|
|
'bulk_deactivate',
|
|
'bulk_update_country',
|
|
] # Enable bulk delete
|
|
|
|
fieldsets = (
|
|
('Keyword Info', {
|
|
'fields': ('keyword', 'industry', 'sector', 'country', 'is_active')
|
|
}),
|
|
('SEO Metrics', {
|
|
'fields': ('volume', 'difficulty')
|
|
}),
|
|
('Timestamps', {
|
|
'fields': ('created_at', 'updated_at')
|
|
}),
|
|
)
|
|
|
|
def has_delete_permission(self, request, obj=None):
|
|
"""Allow deletion for superusers and developers"""
|
|
return request.user.is_superuser or (hasattr(request.user, 'is_developer') and request.user.is_developer())
|
|
|
|
def bulk_activate(self, request, queryset):
|
|
updated = queryset.update(is_active=True)
|
|
self.message_user(request, f'{updated} seed keyword(s) activated.', messages.SUCCESS)
|
|
bulk_activate.short_description = 'Activate selected keywords'
|
|
|
|
def bulk_deactivate(self, request, queryset):
|
|
updated = queryset.update(is_active=False)
|
|
self.message_user(request, f'{updated} seed keyword(s) deactivated.', messages.SUCCESS)
|
|
bulk_deactivate.short_description = 'Deactivate selected keywords'
|
|
|
|
def bulk_update_country(self, request, queryset):
|
|
from django import forms
|
|
|
|
if 'apply' in request.POST:
|
|
country = request.POST.get('country')
|
|
if country:
|
|
updated = queryset.update(country=country)
|
|
self.message_user(request, f'{updated} seed keyword(s) country updated to: {country}', messages.SUCCESS)
|
|
return
|
|
|
|
COUNTRY_CHOICES = [
|
|
('US', 'United States'),
|
|
('GB', 'United Kingdom'),
|
|
('CA', 'Canada'),
|
|
('AU', 'Australia'),
|
|
('IN', 'India'),
|
|
]
|
|
|
|
class CountryForm(forms.Form):
|
|
country = forms.ChoiceField(
|
|
choices=COUNTRY_CHOICES,
|
|
label="Select Country",
|
|
help_text=f"Update country for {queryset.count()} seed keyword(s)"
|
|
)
|
|
|
|
from django.shortcuts import render
|
|
return render(request, 'admin/bulk_action_form.html', {
|
|
'title': 'Update Country',
|
|
'queryset': queryset,
|
|
'form': CountryForm(),
|
|
'action': 'bulk_update_country',
|
|
})
|
|
bulk_update_country.short_description = 'Update country'
|
|
|
|
|
|
class UserResource(resources.ModelResource):
|
|
"""Resource class for exporting Users"""
|
|
class Meta:
|
|
model = User
|
|
fields = ('id', 'email', 'username', 'account__name', 'role',
|
|
'is_active', 'is_staff', 'created_at', 'last_login')
|
|
export_order = fields
|
|
|
|
|
|
@admin.register(User)
|
|
class UserAdmin(ExportMixin, BaseUserAdmin, Igny8ModelAdmin):
|
|
"""
|
|
User admin using both Django's BaseUserAdmin (for user-specific functionality)
|
|
and Unfold's ModelAdmin (for modern UI and styling including popups)
|
|
"""
|
|
resource_class = UserResource
|
|
list_display = ['email', 'username', 'account', 'role', 'is_active', 'is_staff', 'created_at']
|
|
list_filter = ['role', 'account', 'is_active', 'is_staff']
|
|
search_fields = ['email', 'username']
|
|
readonly_fields = ['created_at', 'updated_at']
|
|
|
|
fieldsets = BaseUserAdmin.fieldsets + (
|
|
('IGNY8 Info', {'fields': ('account', 'role')}),
|
|
('Timestamps', {'fields': ('created_at', 'updated_at')}),
|
|
)
|
|
|
|
add_fieldsets = BaseUserAdmin.add_fieldsets + (
|
|
('IGNY8 Info', {'fields': ('account', 'role')}),
|
|
)
|
|
actions = [
|
|
'bulk_set_role_owner',
|
|
'bulk_set_role_admin',
|
|
'bulk_set_role_editor',
|
|
'bulk_set_role_viewer',
|
|
'bulk_activate',
|
|
'bulk_deactivate',
|
|
'bulk_send_password_reset',
|
|
]
|
|
|
|
def get_queryset(self, request):
|
|
"""Filter users by account for non-superusers"""
|
|
qs = super().get_queryset(request)
|
|
if request.user.is_superuser or (hasattr(request.user, 'is_developer') and request.user.is_developer()):
|
|
return qs
|
|
user_account = getattr(request.user, 'account', None)
|
|
if user_account:
|
|
return qs.filter(account=user_account)
|
|
return qs.none()
|
|
|
|
def get_account_display(self, obj):
|
|
"""Safely get account name"""
|
|
try:
|
|
account = getattr(obj, 'account', None)
|
|
return account.name if account else '-'
|
|
except:
|
|
return '-'
|
|
get_account_display.short_description = 'Account'
|
|
|
|
def bulk_set_role_owner(self, request, queryset):
|
|
updated = queryset.update(role='owner')
|
|
self.message_user(request, f'{updated} user(s) role set to Owner.', messages.SUCCESS)
|
|
bulk_set_role_owner.short_description = 'Set role to Owner'
|
|
|
|
def bulk_set_role_admin(self, request, queryset):
|
|
updated = queryset.update(role='admin')
|
|
self.message_user(request, f'{updated} user(s) role set to Admin.', messages.SUCCESS)
|
|
bulk_set_role_admin.short_description = 'Set role to Admin'
|
|
|
|
def bulk_set_role_editor(self, request, queryset):
|
|
updated = queryset.update(role='editor')
|
|
self.message_user(request, f'{updated} user(s) role set to Editor.', messages.SUCCESS)
|
|
bulk_set_role_editor.short_description = 'Set role to Editor'
|
|
|
|
def bulk_set_role_viewer(self, request, queryset):
|
|
updated = queryset.update(role='viewer')
|
|
self.message_user(request, f'{updated} user(s) role set to Viewer.', messages.SUCCESS)
|
|
bulk_set_role_viewer.short_description = 'Set role to Viewer'
|
|
|
|
def bulk_activate(self, request, queryset):
|
|
updated = queryset.update(is_active=True)
|
|
self.message_user(request, f'{updated} user(s) activated.', messages.SUCCESS)
|
|
bulk_activate.short_description = 'Activate users'
|
|
|
|
def bulk_deactivate(self, request, queryset):
|
|
updated = queryset.update(is_active=False)
|
|
self.message_user(request, f'{updated} user(s) deactivated.', messages.SUCCESS)
|
|
bulk_deactivate.short_description = 'Deactivate users'
|
|
|
|
def bulk_send_password_reset(self, request, queryset):
|
|
# TODO: Implement password reset email sending
|
|
count = queryset.count()
|
|
self.message_user(
|
|
request,
|
|
f'{count} password reset email(s) queued for sending. (Email integration required)',
|
|
messages.INFO
|
|
)
|
|
bulk_send_password_reset.short_description = 'Send password reset email'
|
|
|