Compare commits

...

41 Commits

Author SHA1 Message Date
masoodafar-web
2ae6034fbb refactor: remove admin user id from manual payment command
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m13s
2025-12-12 10:22:09 +03:30
masoodafar-web
094846ce8b feat: add manual membership payment processing with wallet and order management
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 7m8s
2025-12-12 10:07:14 +03:30
masoodafar-web
b19cf5e32e fix: use Gregorian week number for API and Persian for display
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m18s
2025-12-12 09:39:20 +03:30
masoodafar-web
b1c3fcfd66 feat: add Persian calendar support for week numbers and dates
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 6m35s
2025-12-12 09:18:31 +03:30
masoodafar-web
1f6c5a1f45 fix: update default sorting from descending to ascending Created
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m43s
2025-12-12 07:49:54 +03:30
masoodafar-web
99b217d5b5 fix: change weekly balance sorting from WeekNumber to Created
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m39s
2025-12-12 07:32:22 +03:30
masoodafar-web
8ee3fe6f7b fix: update week calculation to start from Saturday
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m2s
2025-12-12 07:07:28 +03:30
masoodafar-web
3c7ac68eeb feat: enhance network membership response with detailed stats and hierarchy
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m14s
2025-12-12 06:07:23 +03:30
masoodafar-web
f27418cff4 fix: adjust network tree depth limit and update commission mapping
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m5s
2025-12-12 05:42:19 +03:30
masoodafar-web
aba534e07c fix: update week calculation to use Saturday as the start of the week
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m12s
2025-12-12 04:37:34 +03:30
masoodafar-web
af3a29ed27 fix: adjust week calculation to FirstDay and Saturday
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 3m9s
2025-12-12 04:23:14 +03:30
masoodafar-web
db951699f8 fix: handle null pool and adjust week calculation rules
Some checks failed
Build and Deploy to Kubernetes / build-and-deploy (push) Has been cancelled
2025-12-12 04:22:16 +03:30
masoodafar-web
bb6b7c709c feat: add search text filter for user query
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m40s
2025-12-12 03:18:18 +03:30
masoodafar-web
12749ccb01 chore: bump version to 0.0.146
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 11m40s
2025-12-12 01:41:48 +03:30
masoodafar-web
ff1c1d5d61 feat: add IsActive field to UserClubFeatures for admin management 2025-12-12 01:40:26 +03:30
masoodafar-web
aa66ca10c8 feat: add Mapster mapping and update network tree request
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m11s
2025-12-08 04:10:46 +03:30
masoud
69d5ecf0d7 revert: Back to simple MigrateAsync - it's already idempotent
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m5s
MigrateAsync is smart enough to:
- Create DB if it doesn't exist
- Apply only pending migrations
- Skip if DB is up to date

The orphaned CMS.mdf files were cleaned from SQL Server.
2025-12-07 21:27:46 +00:00
masoud
1dbe90d020 fix: Use EnsureCreatedAsync instead of MigrateAsync for initial db setup
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m9s
- EnsureCreatedAsync creates db only if it doesn't exist
- Returns false if db already exists (idempotent)
- Prevents 'CMS.mdf already exists' error
- Still applies pending migrations when needed
2025-12-07 21:21:47 +00:00
masoud
7ec9c4077c fix: Improve database migration strategy - only migrate if needed
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m16s
- Check if database exists before creating
- Only apply pending migrations
- Skip migration if database is up to date
- Prevents 'file already exists' error on restart
2025-12-07 21:16:49 +00:00
masoud
70a06933a9 fix: update kubeconfig for deployment (127.0.0.1 → 194.5.195.53)
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m21s
2025-12-07 19:16:58 +00:00
masoud
d51ef47c16 docs: add multi-remote push note
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m15s
2025-12-07 18:50:16 +00:00
masoud
d9ccfcc587 fix: replace tabs with spaces in workflow file
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 3m33s
2025-12-07 18:00:58 +00:00
masoud
9e3e0c68c2 trigger: rebuild with HTTP/2 config 2025-12-07 18:00:01 +00:00
masoud
6f3bb3e2c3 chore: trigger workflow for HTTP/2 deployment 2025-12-07 17:58:13 +00:00
masoud
0fe39cf9f6 feat: enable pure HTTP/2 for gRPC compatibility with nginx ingress 2025-12-07 17:53:43 +00:00
masoud
e5fc13b9e2 chore: migrate from IP to domain (git.foursat.afrino.co) 2025-12-07 17:02:17 +00:00
masoodafar-web
e7937fe7ee fix: Remove trailing whitespace in Seq config 2025-12-07 20:09:01 +03:30
masoodafar-web
9fb5119fc1 fix: Remove trailing whitespace in Seq config 2025-12-07 19:46:36 +03:30
masoodafar-web
4d6d0dfc9a feat: Add kubectl deployment and image rollout steps 2025-12-06 22:57:25 +03:30
masoodafar-web
adcc776230 feat: Add Kestrel HTTP/2 configuration for macOS 2025-12-06 22:24:28 +03:30
masoodafar-web
a0da7eb8e0 feat: Implement real Daya API integration with configurable mock/real service 2025-12-06 21:02:51 +03:30
masoodafar-web
e46f54ca5b Merge branch 'test' into kub-stage
All checks were successful
Build and Deploy to Kubernetes / build-and-deploy (push) Successful in 2m16s
2025-12-06 00:17:39 +03:30
masoodafar-web
00ba911703 feat: Add archive and schedule fields to PublicMessages 2025-12-06 00:16:30 +03:30
masoud
bfcb0d880f feat: Read Seq logging config from appsettings.json + add Console logging 2025-12-05 19:32:28 +00:00
masoud
ed642947dc Fix: Suppress EF Core 9 PendingModelChangesWarning 2025-12-05 18:55:12 +00:00
masoud
c8db56ab2b Retry 2025-12-05 18:11:53 +00:00
masoud
80f4e124d6 Add insecure registry 2025-12-05 18:11:14 +00:00
masoud
e17b7f4588 Retry with fixed Dockerfile 2025-12-05 18:09:17 +00:00
masoud
4438a4d248 Fix Dockerfile for buildkit 2025-12-05 18:07:39 +00:00
masoud
eb43b084b2 Trigger 2025-12-05 18:06:11 +00:00
masoud
0c054739ba Fix proxy password 2025-12-05 18:05:52 +00:00
107 changed files with 16029 additions and 420 deletions

View File

@@ -6,7 +6,7 @@ on:
- kub-stage
env:
REGISTRY: 194.5.195.53:30080
REGISTRY: git.foursat.afrino.co
IMAGE_NAME: admin/cms
jobs:
@@ -16,26 +16,34 @@ jobs:
image: docker:latest
options: --privileged
env:
HTTP_PROXY: http://proxyuser:87zH26nbqT@46.249.98.211:3128
HTTPS_PROXY: http://proxyuser:87zH26nbqT@46.249.98.211:3128
HTTP_PROXY: http://proxyuser:87zH26nbqT2@46.249.98.211:3128
HTTPS_PROXY: http://proxyuser:87zH26nbqT2@46.249.98.211:3128
NO_PROXY: localhost,127.0.0.1,gitea-svc,194.5.195.53,10.0.0.0/8
steps:
- name: Install git
run: apk add --no-cache git
- name: Clone repo
- name: Install dependencies
run: |
git clone --depth 1 --branch kub-stage http://gitea-svc:3000/admin/CMS.git .
apk add --no-cache git curl
# Install kubectl
curl -LO "https://dl.k8s.io/release/$(curl -L -s https://dl.k8s.io/release/stable.txt)/bin/linux/amd64/kubectl"
chmod +x kubectl
mv kubectl /usr/local/bin/
- name: Start Docker daemon with proxy
- name: Start Docker daemon with insecure registry
run: |
mkdir -p /etc/docker
cat > /etc/docker/daemon.json << 'DAEMON'
{
"insecure-registries": ["git.foursat.afrino.co", "gitea-svc:3000"]
}
DAEMON
mkdir -p ~/.docker
cat > ~/.docker/config.json << CONF
cat > ~/.docker/config.json << 'CONF'
{
"proxies": {
"default": {
"httpProxy": "http://proxyuser:87zH26nbqT@46.249.98.211:3128",
"httpsProxy": "http://proxyuser:87zH26nbqT@46.249.98.211:3128",
"httpProxy": "http://proxyuser:87zH26nbqT2@46.249.98.211:3128",
"httpsProxy": "http://proxyuser:87zH26nbqT2@46.249.98.211:3128",
"noProxy": "localhost,127.0.0.1,gitea-svc,194.5.195.53,10.0.0.0/8"
}
}
@@ -46,13 +54,17 @@ jobs:
docker info >/dev/null 2>&1 && break || sleep 2
done
docker info
- name: Checkout code
run: |
git clone --depth 1 --branch kub-stage http://gitea-svc:3000/admin/CMS.git .
git log -1 --format="%H %s"
- name: Build Docker Image
run: |
docker build -t ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${{ github.sha }} \
-t ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:latest \
--build-arg HTTP_PROXY=http://proxyuser:87zH26nbqT@46.249.98.211:3128 \
--build-arg HTTPS_PROXY=http://proxyuser:87zH26nbqT@46.249.98.211:3128 \
--build-arg HTTP_PROXY=http://proxyuser:87zH26nbqT2@46.249.98.211:3128 \
--build-arg HTTPS_PROXY=http://proxyuser:87zH26nbqT2@46.249.98.211:3128 \
.
- name: Push to Registry
@@ -60,3 +72,15 @@ jobs:
echo "${{ secrets.REGISTRY_PASSWORD }}" | docker login ${{ env.REGISTRY }} -u admin --password-stdin
docker push ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${{ github.sha }}
docker push ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:latest
- name: Deploy to Kubernetes
run: |
# Setup kubeconfig
mkdir -p ~/.kube
echo "${{ secrets.KUBECONFIG }}" | base64 -d > ~/.kube/config
# Restart deployment to pull new image
kubectl rollout restart deployment/cms || echo "Deployment doesn't exist yet"
# Wait for rollout to complete
kubectl rollout status deployment/cms --timeout=5m || echo "Deployment rollout pending"

1
.trigger Normal file
View File

@@ -0,0 +1 @@
# Trigger

View File

@@ -1,11 +1,12 @@
FROM mcr.microsoft.com/dotnet/sdk:9.0 AS build
WORKDIR /src
COPY src/*.sln ./ 2>/dev/null || true
COPY src/*/*.csproj ./
RUN for file in *.csproj; do mkdir -p "${file%.*}" && mv "$file" "${file%.*}/"; done 2>/dev/null || true
RUN dotnet restore "CMSMicroservice.WebApi/CMSMicroservice.WebApi.csproj" || dotnet restore
# Copy solution and project files
COPY src/ ./
RUN dotnet publish "CMSMicroservice.WebApi/CMSMicroservice.WebApi.csproj" -c Release -o /app/publish
# Restore and publish
RUN dotnet restore "CMSMicroservice.WebApi/CMSMicroservice.WebApi.csproj"
RUN dotnet publish "CMSMicroservice.WebApi/CMSMicroservice.WebApi.csproj" -c Release -o /app/publish --no-restore
FROM mcr.microsoft.com/dotnet/aspnet:9.0 AS runtime
WORKDIR /app

View File

@@ -256,3 +256,4 @@ curl http://localhost:5133/health/live # Liveness probe (K8s)
## 📝 License
Proprietary - FourSat Company
# Multi-remote push enabled

View File

@@ -0,0 +1,490 @@
# Club Feature Management Services - Implementation Guide
## Overview
Admin services for managing user club features (enable/disable features per user).
## Created Files
### 1. CQRS Layer (Application)
#### Query: GetUserClubFeatures
**Location:** `/CMS/src/CMSMicroservice.Application/ClubFeatureCQ/Queries/GetUserClubFeatures/`
**Files:**
- `GetUserClubFeaturesQuery.cs` - Query definition
- `GetUserClubFeaturesQueryHandler.cs` - Query handler
- `UserClubFeatureDto.cs` - Response DTO
**Purpose:** Get list of all club features for a specific user with their active status.
**Input:**
```csharp
public record GetUserClubFeaturesQuery : IRequest<List<UserClubFeatureDto>>
{
public long UserId { get; init; }
}
```
**Output:**
```csharp
public class UserClubFeatureDto
{
public long Id { get; set; }
public long UserId { get; set; }
public long ClubMembershipId { get; set; }
public long ClubFeatureId { get; set; }
public string FeatureTitle { get; set; }
public string? FeatureDescription { get; set; }
public bool IsActive { get; set; }
public DateTime GrantedAt { get; set; }
public string? Notes { get; set; }
}
```
**Logic:**
- Joins `UserClubFeatures` with `ClubFeature` table
- Filters by `UserId` and `!IsDeleted`
- Returns list of features with their active status
---
#### Command: ToggleUserClubFeature
**Location:** `/CMS/src/CMSMicroservice.Application/ClubFeatureCQ/Commands/ToggleUserClubFeature/`
**Files:**
- `ToggleUserClubFeatureCommand.cs` - Command definition
- `ToggleUserClubFeatureCommandHandler.cs` - Command handler
- `ToggleUserClubFeatureResponse.cs` - Response DTO
**Purpose:** Enable or disable a specific club feature for a user.
**Input:**
```csharp
public record ToggleUserClubFeatureCommand : IRequest<ToggleUserClubFeatureResponse>
{
public long UserId { get; init; }
public long ClubFeatureId { get; init; }
public bool IsActive { get; init; }
}
```
**Output:**
```csharp
public class ToggleUserClubFeatureResponse
{
public bool Success { get; set; }
public string Message { get; set; }
public long? UserClubFeatureId { get; set; }
public bool? IsActive { get; set; }
}
```
**Validations:**
1. ✅ User exists and not deleted
2. ✅ Club feature exists and not deleted
3. ✅ User has this feature assigned (exists in UserClubFeatures)
**Logic:**
- Find `UserClubFeature` record by `UserId` + `ClubFeatureId`
- Update `IsActive` field
- Set `LastModified` timestamp
- Save changes
**Error Messages:**
- "کاربر یافت نشد" - User not found
- "ویژگی باشگاه یافت نشد" - Club feature not found
- "این ویژگی برای کاربر یافت نشد" - User doesn't have this feature
**Success Messages:**
- "ویژگی با موفقیت فعال شد" - Feature activated successfully
- "ویژگی با موفقیت غیرفعال شد" - Feature deactivated successfully
---
### 2. gRPC Layer (Protobuf + WebApi)
#### Proto Definition
**File:** `/CMS/src/CMSMicroservice.Protobuf/Protos/clubmembership.proto`
**Added RPC Methods:**
```protobuf
rpc GetUserClubFeatures(GetUserClubFeaturesRequest) returns (GetUserClubFeaturesResponse){
option (google.api.http) = {
get: "/ClubFeature/GetUserFeatures"
};
};
rpc ToggleUserClubFeature(ToggleUserClubFeatureRequest) returns (ToggleUserClubFeatureResponse){
option (google.api.http) = {
post: "/ClubFeature/ToggleFeature"
body: "*"
};
};
```
**Message Definitions:**
```protobuf
message GetUserClubFeaturesRequest {
int64 user_id = 1;
}
message GetUserClubFeaturesResponse {
repeated UserClubFeatureModel features = 1;
}
message UserClubFeatureModel {
int64 id = 1;
int64 user_id = 2;
int64 club_membership_id = 3;
int64 club_feature_id = 4;
string feature_title = 5;
string feature_description = 6;
bool is_active = 7;
google.protobuf.Timestamp granted_at = 8;
string notes = 9;
}
message ToggleUserClubFeatureRequest {
int64 user_id = 1;
int64 club_feature_id = 2;
bool is_active = 3;
}
message ToggleUserClubFeatureResponse {
bool success = 1;
string message = 2;
google.protobuf.Int64Value user_club_feature_id = 3;
google.protobuf.BoolValue is_active = 4;
}
```
---
#### gRPC Service Implementation
**File:** `/CMS/src/CMSMicroservice.WebApi/Services/ClubMembershipService.cs`
**Added Methods:**
```csharp
public override async Task<GetUserClubFeaturesResponse> GetUserClubFeatures(
GetUserClubFeaturesRequest request,
ServerCallContext context)
{
return await _dispatchRequestToCQRS.Handle<
GetUserClubFeaturesRequest,
GetUserClubFeaturesQuery,
GetUserClubFeaturesResponse>(request, context);
}
public override async Task<Protobuf.Protos.ClubMembership.ToggleUserClubFeatureResponse>
ToggleUserClubFeature(
ToggleUserClubFeatureRequest request,
ServerCallContext context)
{
return await _dispatchRequestToCQRS.Handle<
ToggleUserClubFeatureRequest,
ToggleUserClubFeatureCommand,
Protobuf.Protos.ClubMembership.ToggleUserClubFeatureResponse>(request, context);
}
```
---
#### AutoMapper Profile
**File:** `/CMS/src/CMSMicroservice.WebApi/Common/Mappings/ClubFeatureProfile.cs`
**Mappings:**
1. `GetUserClubFeaturesRequest``GetUserClubFeaturesQuery`
2. `UserClubFeatureDto``UserClubFeatureModel` (Proto)
3. `List<UserClubFeatureDto>``GetUserClubFeaturesResponse`
4. `ToggleUserClubFeatureRequest``ToggleUserClubFeatureCommand`
5. `ToggleUserClubFeatureResponse` (App) → `ToggleUserClubFeatureResponse` (Proto)
**Special Handling:**
- DateTime conversion to `Timestamp` (Protobuf format)
- Null-safe mapping for optional fields
- Fully qualified type names to avoid ambiguity
---
## API Endpoints
### 1. Get User Club Features
**Method:** GET
**Endpoint:** `/ClubFeature/GetUserFeatures`
**Request:**
```json
{
"user_id": 123
}
```
**Response:**
```json
{
"features": [
{
"id": 1,
"user_id": 123,
"club_membership_id": 456,
"club_feature_id": 1,
"feature_title": "دسترسی به فروشگاه تخفیف",
"feature_description": "امکان خرید از فروشگاه تخفیف",
"is_active": true,
"granted_at": "2025-12-09T18:30:00Z",
"notes": "اعطا شده به‌طور خودکار هنگام فعالسازی"
}
]
}
```
---
### 2. Toggle User Club Feature
**Method:** POST
**Endpoint:** `/ClubFeature/ToggleFeature`
**Request:**
```json
{
"user_id": 123,
"club_feature_id": 1,
"is_active": false
}
```
**Response (Success):**
```json
{
"success": true,
"message": "ویژگی با موفقیت غیرفعال شد",
"user_club_feature_id": 1,
"is_active": false
}
```
**Response (Error - User Not Found):**
```json
{
"success": false,
"message": "کاربر یافت نشد"
}
```
**Response (Error - Feature Not Found):**
```json
{
"success": false,
"message": "ویژگی باشگاه یافت نشد"
}
```
**Response (Error - User Doesn't Have Feature):**
```json
{
"success": false,
"message": "این ویژگی برای کاربر یافت نشد"
}
```
---
## Database Schema
### Table: UserClubFeatures
Existing table with newly added `IsActive` field:
```sql
CREATE TABLE [CMS].[UserClubFeatures]
(
[Id] BIGINT IDENTITY(1,1) PRIMARY KEY,
[UserId] BIGINT NOT NULL,
[ClubMembershipId] BIGINT NOT NULL,
[ClubFeatureId] BIGINT NOT NULL,
[GrantedAt] DATETIME2 NOT NULL,
[IsActive] BIT NOT NULL DEFAULT 1, -- ← NEW FIELD
[Notes] NVARCHAR(MAX) NULL,
[Created] DATETIME2 NOT NULL,
[CreatedBy] NVARCHAR(MAX) NULL,
[LastModified] DATETIME2 NULL,
[LastModifiedBy] NVARCHAR(MAX) NULL,
[IsDeleted] BIT NOT NULL DEFAULT 0,
CONSTRAINT FK_UserClubFeatures_Users FOREIGN KEY ([UserId])
REFERENCES [Identity].[Users]([Id]),
CONSTRAINT FK_UserClubFeatures_ClubMembership FOREIGN KEY ([ClubMembershipId])
REFERENCES [CMS].[ClubMembership]([Id]),
CONSTRAINT FK_UserClubFeatures_ClubFeatures FOREIGN KEY ([ClubFeatureId])
REFERENCES [CMS].[ClubFeatures]([Id])
);
```
---
## Usage Examples
### Admin Panel Scenario
#### 1. View User's Club Features
```csharp
// Admin selects user ID: 123
var request = new GetUserClubFeaturesRequest { UserId = 123 };
var response = await client.GetUserClubFeaturesAsync(request);
// Display in grid:
foreach (var feature in response.Features)
{
Console.WriteLine($"Feature: {feature.FeatureTitle}");
Console.WriteLine($"Status: {(feature.IsActive ? "فعال" : "غیرفعال")}");
Console.WriteLine($"Granted: {feature.GrantedAt}");
Console.WriteLine("---");
}
```
**Output:**
```
Feature: دسترسی به فروشگاه تخفیف
Status: فعال
Granted: 2025-12-09 18:30:00
---
Feature: دسترسی به کمیسیون هفتگی
Status: فعال
Granted: 2025-12-09 18:30:00
---
Feature: دسترسی به شارژ شبکه
Status: غیرفعال
Granted: 2025-12-09 18:30:00
---
```
---
#### 2. Disable a Feature
```csharp
// Admin clicks "Disable" on Feature ID: 3
var request = new ToggleUserClubFeatureRequest
{
UserId = 123,
ClubFeatureId = 3,
IsActive = false
};
var response = await client.ToggleUserClubFeatureAsync(request);
if (response.Success)
{
Console.WriteLine(response.Message);
// Output: ویژگی با موفقیت غیرفعال شد
}
```
---
#### 3. Re-enable a Feature
```csharp
// Admin clicks "Enable" on Feature ID: 3
var request = new ToggleUserClubFeatureRequest
{
UserId = 123,
ClubFeatureId = 3,
IsActive = true
};
var response = await client.ToggleUserClubFeatureAsync(request);
if (response.Success)
{
Console.WriteLine(response.Message);
// Output: ویژگی با موفقیت فعال شد
}
```
---
## Testing Checklist
### Unit Tests (Recommended)
- [ ] GetUserClubFeaturesQueryHandler returns correct DTOs
- [ ] ToggleUserClubFeatureCommandHandler validates user exists
- [ ] ToggleUserClubFeatureCommandHandler validates feature exists
- [ ] ToggleUserClubFeatureCommandHandler validates user has feature
- [ ] ToggleUserClubFeatureCommandHandler updates IsActive correctly
- [ ] ToggleUserClubFeatureCommandHandler sets LastModified timestamp
### Integration Tests
- [ ] gRPC GetUserClubFeatures endpoint returns data
- [ ] gRPC ToggleUserClubFeature endpoint updates database
- [ ] AutoMapper mappings work correctly
- [ ] Proto serialization/deserialization works
### Manual Testing
1. **Get Features:**
```bash
grpcurl -d '{"user_id": 123}' \
-plaintext localhost:5000 \
clubmembership.ClubMembershipContract/GetUserClubFeatures
```
2. **Disable Feature:**
```bash
grpcurl -d '{"user_id": 123, "club_feature_id": 1, "is_active": false}' \
-plaintext localhost:5000 \
clubmembership.ClubMembershipContract/ToggleUserClubFeature
```
3. **Verify in Database:**
```sql
SELECT Id, UserId, ClubFeatureId, IsActive, LastModified
FROM CMS.UserClubFeatures
WHERE UserId = 123;
```
---
## Build Status
✅ **All projects build successfully**
- CMSMicroservice.Domain: ✅
- CMSMicroservice.Application: ✅ (0 errors, 274 warnings)
- CMSMicroservice.Protobuf: ✅
- CMSMicroservice.WebApi: ✅ (0 errors, 17 warnings)
---
## Next Steps (Optional Enhancements)
1. **Authorization:**
- Add `[Authorize(Roles = "Admin")]` attribute
- Validate admin permissions before toggling
2. **Audit Logging:**
- Log who changed the feature status
- Track `LastModifiedBy` field
3. **Bulk Operations:**
- Add endpoint to toggle multiple features at once
- Add endpoint to enable/disable all features for a user
4. **History Tracking:**
- Create `UserClubFeatureHistory` table
- Log every status change with timestamp and reason
5. **Notifications:**
- Send notification to user when feature is disabled
- Email/SMS alert for important features
6. **Business Rules:**
- Add validation: prevent disabling critical features
- Add expiration dates for features
- Add feature dependencies (e.g., Feature B requires Feature A)
---
## Summary
✅ Created CQRS Query + Command for club feature management
✅ Created gRPC Proto definitions and services
✅ Created AutoMapper mappings
✅ All builds successful
✅ Ready for deployment and testing
**Total Files Created:** 8
**Total Lines of Code:** ~350
**Build Errors:** 0
**Status:** ✅ Complete and ready for use

View File

@@ -0,0 +1,24 @@
using MediatR;
namespace CMSMicroservice.Application.ClubFeatureCQ.Commands.ToggleUserClubFeature;
/// <summary>
/// کامند برای فعال/غیرفعال کردن یک ویژگی باشگاه برای کاربر
/// </summary>
public record ToggleUserClubFeatureCommand : IRequest<ToggleUserClubFeatureResponse>
{
/// <summary>
/// شناسه کاربر
/// </summary>
public long UserId { get; init; }
/// <summary>
/// شناسه ویژگی باشگاه
/// </summary>
public long ClubFeatureId { get; init; }
/// <summary>
/// وضعیت مورد نظر (فعال/غیرفعال)
/// </summary>
public bool IsActive { get; init; }
}

View File

@@ -0,0 +1,78 @@
using CMSMicroservice.Application.Common.Interfaces;
using MediatR;
using Microsoft.EntityFrameworkCore;
namespace CMSMicroservice.Application.ClubFeatureCQ.Commands.ToggleUserClubFeature;
/// <summary>
/// هندلر برای فعال/غیرفعال کردن ویژگی باشگاه کاربر
/// </summary>
public class ToggleUserClubFeatureCommandHandler : IRequestHandler<ToggleUserClubFeatureCommand, ToggleUserClubFeatureResponse>
{
private readonly IApplicationDbContext _context;
public ToggleUserClubFeatureCommandHandler(IApplicationDbContext context)
{
_context = context;
}
public async Task<ToggleUserClubFeatureResponse> Handle(ToggleUserClubFeatureCommand request, CancellationToken cancellationToken)
{
// بررسی وجود کاربر
var userExists = await _context.Users
.AnyAsync(u => u.Id == request.UserId && !u.IsDeleted, cancellationToken);
if (!userExists)
{
return new ToggleUserClubFeatureResponse
{
Success = false,
Message = "کاربر یافت نشد"
};
}
// بررسی وجود ویژگی
var featureExists = await _context.ClubFeatures
.AnyAsync(cf => cf.Id == request.ClubFeatureId && !cf.IsDeleted, cancellationToken);
if (!featureExists)
{
return new ToggleUserClubFeatureResponse
{
Success = false,
Message = "ویژگی باشگاه یافت نشد"
};
}
// یافتن رکورد UserClubFeature
var userClubFeature = await _context.UserClubFeatures
.FirstOrDefaultAsync(
ucf => ucf.UserId == request.UserId
&& ucf.ClubFeatureId == request.ClubFeatureId
&& !ucf.IsDeleted,
cancellationToken);
if (userClubFeature == null)
{
return new ToggleUserClubFeatureResponse
{
Success = false,
Message = "این ویژگی برای کاربر یافت نشد"
};
}
// به‌روزرسانی وضعیت
userClubFeature.IsActive = request.IsActive;
userClubFeature.LastModified = DateTime.UtcNow;
await _context.SaveChangesAsync(cancellationToken);
return new ToggleUserClubFeatureResponse
{
Success = true,
Message = request.IsActive ? "ویژگی با موفقیت فعال شد" : "ویژگی با موفقیت غیرفعال شد",
UserClubFeatureId = userClubFeature.Id,
IsActive = userClubFeature.IsActive
};
}
}

View File

@@ -0,0 +1,27 @@
namespace CMSMicroservice.Application.ClubFeatureCQ.Commands.ToggleUserClubFeature;
/// <summary>
/// پاسخ کامند فعال/غیرفعال کردن ویژگی باشگاه
/// </summary>
public class ToggleUserClubFeatureResponse
{
/// <summary>
/// موفقیت عملیات
/// </summary>
public bool Success { get; set; }
/// <summary>
/// پیام
/// </summary>
public string Message { get; set; }
/// <summary>
/// شناسه رکورد UserClubFeature
/// </summary>
public long? UserClubFeatureId { get; set; }
/// <summary>
/// وضعیت جدید
/// </summary>
public bool? IsActive { get; set; }
}

View File

@@ -0,0 +1,14 @@
using MediatR;
namespace CMSMicroservice.Application.ClubFeatureCQ.Queries.GetUserClubFeatures;
/// <summary>
/// دریافت لیست ویژگی‌های باشگاه اختصاص یافته به کاربر
/// </summary>
public record GetUserClubFeaturesQuery : IRequest<List<UserClubFeatureDto>>
{
/// <summary>
/// شناسه کاربر
/// </summary>
public long UserId { get; init; }
}

View File

@@ -0,0 +1,40 @@
using CMSMicroservice.Application.Common.Interfaces;
using MediatR;
using Microsoft.EntityFrameworkCore;
namespace CMSMicroservice.Application.ClubFeatureCQ.Queries.GetUserClubFeatures;
/// <summary>
/// هندلر برای دریافت لیست ویژگی‌های باشگاه یک کاربر
/// </summary>
public class GetUserClubFeaturesQueryHandler : IRequestHandler<GetUserClubFeaturesQuery, List<UserClubFeatureDto>>
{
private readonly IApplicationDbContext _context;
public GetUserClubFeaturesQueryHandler(IApplicationDbContext context)
{
_context = context;
}
public async Task<List<UserClubFeatureDto>> Handle(GetUserClubFeaturesQuery request, CancellationToken cancellationToken)
{
var userClubFeatures = await _context.UserClubFeatures
.Include(ucf => ucf.ClubFeature)
.Where(ucf => ucf.UserId == request.UserId && !ucf.IsDeleted)
.Select(ucf => new UserClubFeatureDto
{
Id = ucf.Id,
UserId = ucf.UserId,
ClubMembershipId = ucf.ClubMembershipId,
ClubFeatureId = ucf.ClubFeatureId,
FeatureTitle = ucf.ClubFeature.Title,
FeatureDescription = ucf.ClubFeature.Description,
IsActive = ucf.IsActive,
GrantedAt = ucf.GrantedAt,
Notes = ucf.Notes
})
.ToListAsync(cancellationToken);
return userClubFeatures;
}
}

View File

@@ -0,0 +1,52 @@
namespace CMSMicroservice.Application.ClubFeatureCQ.Queries.GetUserClubFeatures;
/// <summary>
/// DTO برای ویژگی‌های باشگاه کاربر
/// </summary>
public class UserClubFeatureDto
{
/// <summary>
/// شناسه رکورد UserClubFeature
/// </summary>
public long Id { get; set; }
/// <summary>
/// شناسه کاربر
/// </summary>
public long UserId { get; set; }
/// <summary>
/// شناسه عضویت باشگاه
/// </summary>
public long ClubMembershipId { get; set; }
/// <summary>
/// شناسه ویژگی باشگاه
/// </summary>
public long ClubFeatureId { get; set; }
/// <summary>
/// عنوان ویژگی
/// </summary>
public string FeatureTitle { get; set; }
/// <summary>
/// توضیحات ویژگی
/// </summary>
public string? FeatureDescription { get; set; }
/// <summary>
/// وضعیت فعال/غیرفعال ویژگی برای این کاربر
/// </summary>
public bool IsActive { get; set; }
/// <summary>
/// تاریخ اعطای ویژگی
/// </summary>
public DateTime GrantedAt { get; set; }
/// <summary>
/// یادداشت
/// </summary>
public string? Notes { get; set; }
}

View File

@@ -3,11 +3,13 @@ using CMSMicroservice.Application.Common.Interfaces;
using CMSMicroservice.Application.Common.Models;
using CMSMicroservice.Domain.Entities;
using CMSMicroservice.Domain.Entities.Club;
using CMSMicroservice.Domain.Entities.Commission;
using CMSMicroservice.Domain.Entities.History;
using CMSMicroservice.Domain.Enums;
using MediatR;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Logging;
using System.Globalization;
namespace CMSMicroservice.Application.ClubMembershipCQ.Commands.ActivateClubMembership;
@@ -135,9 +137,15 @@ public class ActivateClubMembershipCommandHandler : IRequestHandler<ActivateClub
.FirstOrDefaultAsync(
c => c.Key == "Club.MembershipGiftValue" && c.IsActive,
cancellationToken
);
var activationFeeConfig = await _context.SystemConfigurations
.FirstOrDefaultAsync(
c => c.Key == "Club.ActivationFee" && c.IsActive,
cancellationToken
);
long giftValue = 25_200_000; // مقدار پیش‌فرض
long giftValue = 28_000_000; // مقدار پیش‌فرض
if (giftValueConfig != null && long.TryParse(giftValueConfig.Value, out var configValue))
{
giftValue = configValue;
@@ -152,11 +160,27 @@ public class ActivateClubMembershipCommandHandler : IRequestHandler<ActivateClub
"Club.MembershipGiftValue not found in configuration, using default: {GiftValue}",
giftValue
);
}
long activationFeeValue = 25_200_000; // مقدار پیش‌فرض
if (activationFeeConfig != null && long.TryParse(activationFeeConfig.Value, out var activationFeeConfigValue))
{
activationFeeValue = activationFeeConfigValue;
_logger.LogInformation(
"Using Club.ActivationFee from configuration: {activationFeeValue}",
activationFeeValue
);
}
else
{
_logger.LogWarning(
"Club.ActivationFee not found in configuration, using default: {activationFeeValue}",
activationFeeValue
);
}
ClubMembership entity;
bool isNewMembership = existingMembership == null;
var activationDate = DateTime.UtcNow;
var activationDate = DateTime.Now;
if (isNewMembership)
{
@@ -166,7 +190,7 @@ public class ActivateClubMembershipCommandHandler : IRequestHandler<ActivateClub
UserId = user.Id,
IsActive = true,
ActivatedAt = activationDate,
InitialContribution = 56_000_000,
InitialContribution =activationFeeValue,
GiftValue = giftValue, // مقدار از تنظیمات
TotalEarned = 0,
PurchaseMethod = user.PackagePurchaseMethod
@@ -225,6 +249,83 @@ public class ActivateClubMembershipCommandHandler : IRequestHandler<ActivateClub
_context.ClubMembershipHistories.Add(history);
await _context.SaveChangesAsync(cancellationToken);
// ⭐ 8. اضافه کردن مبلغ به Pool هفته جاری
var currentWeekNumber = GetCurrentWeekNumber();
var weeklyPool = await _context.WeeklyCommissionPools
.FirstOrDefaultAsync(p => p.WeekNumber == currentWeekNumber, cancellationToken);
if (weeklyPool == null)
{
// ایجاد Pool جدید برای این هفته
weeklyPool = new WeeklyCommissionPool
{
WeekNumber = currentWeekNumber,
TotalPoolAmount = activationFeeValue, // مبلغ هدیه به Pool اضافه میشه
TotalBalances = 0, // در CalculateWeeklyBalances محاسبه میشه
ValuePerBalance = 0, // در CalculateWeeklyCommissionPool محاسبه میشه
IsCalculated = false,
CalculatedAt = null
};
await _context.WeeklyCommissionPools.AddAsync(weeklyPool, cancellationToken);
_logger.LogInformation(
"Created new WeeklyCommissionPool for {WeekNumber} with initial amount: {Amount}",
currentWeekNumber,
giftValue
);
}
else
{
// اضافه کردن به Pool موجود
weeklyPool.TotalPoolAmount += activationFeeValue;
_context.WeeklyCommissionPools.Update(weeklyPool);
_logger.LogInformation(
"Added {Amount} to existing WeeklyCommissionPool for {WeekNumber}. New total: {NewTotal}",
activationFeeValue,
currentWeekNumber,
weeklyPool.TotalPoolAmount
);
}
await _context.SaveChangesAsync(cancellationToken);
// 9. اضافه کردن ویژگی‌های باشگاه برای کاربر (فقط برای عضویت جدید)
if (isNewMembership)
{
var clubFeatures = await _context.ClubFeatures
.Where(f => !f.IsDeleted && new long[] { 1, 2, 3, 4 }.Contains(f.Id))
.ToListAsync(cancellationToken);
if (clubFeatures.Any())
{
var userClubFeatures = clubFeatures.Select(feature => new UserClubFeature
{
UserId = user.Id,
ClubMembershipId = entity.Id,
ClubFeatureId = feature.Id,
GrantedAt = activationDate,
IsActive = true,
Notes = "اعطا شده به‌طور خودکار هنگام فعالسازی"
}).ToList(); _context.UserClubFeatures.AddRange(userClubFeatures);
await _context.SaveChangesAsync(cancellationToken);
_logger.LogInformation(
"Granted {Count} club features to UserId {UserId}",
clubFeatures.Count,
user.Id
);
}
else
{
_logger.LogWarning(
"No club features found to grant to UserId {UserId}",
user.Id
);
}
}
_logger.LogInformation(
"Club membership activated successfully. UserId: {UserId}, MembershipId: {MembershipId}",
user.Id,
@@ -243,4 +344,15 @@ public class ActivateClubMembershipCommandHandler : IRequestHandler<ActivateClub
throw;
}
}
/// <summary>
/// دریافت شماره هفته جاری (مثال: "2025-W50")
/// </summary>
private string GetCurrentWeekNumber()
{
var now = DateTime.Now;
var calendar = CultureInfo.CurrentCulture.Calendar;
var weekOfYear = calendar.GetWeekOfYear(now, CalendarWeekRule.FirstDay, DayOfWeek.Saturday);
return $"{now.Year}-W{weekOfYear:D2}";
}
}

View File

@@ -54,7 +54,7 @@ public class AssignClubFeatureCommandHandler : IRequestHandler<AssignClubFeature
UserId = request.UserId,
ClubMembershipId = membership.Id,
ClubFeatureId = request.FeatureId,
GrantedAt = request.GrantedAt ?? DateTime.UtcNow,
GrantedAt = request.GrantedAt ?? DateTime.Now,
Notes = request.Notes
};

View File

@@ -25,7 +25,7 @@ public class GetClubMembershipHistoryQueryHandler : IRequestHandler<GetClubMembe
query = query.Where(x => x.UserId == request.UserId.Value);
}
query = query.ApplyOrder(sortBy: request.SortBy ?? "-Created"); // پیش‌فرض: جدیدترین اول
query = query.ApplyOrder(sortBy: request.SortBy ?? "Created"); // پیش‌فرض: جدیدترین اول
var meta = await query.GetMetaData(request.PaginationState, cancellationToken);

View File

@@ -11,7 +11,7 @@ public class GetClubStatisticsQueryHandler : IRequestHandler<GetClubStatisticsQu
public async Task<GetClubStatisticsResponseDto> Handle(GetClubStatisticsQuery request, CancellationToken cancellationToken)
{
var now = DateTime.UtcNow;
var now = DateTime.Now;
// Basic statistics
var totalMembers = await _context.ClubMemberships.CountAsync(cancellationToken);

View File

@@ -33,10 +33,10 @@ public class ApproveWithdrawalCommandHandler : IRequestHandler<ApproveWithdrawal
// Update status to Withdrawn (approved)
payout.Status = CommissionPayoutStatus.Withdrawn;
payout.WithdrawnAt = DateTime.UtcNow;
payout.WithdrawnAt = DateTime.Now;
payout.ProcessedBy = _currentUser.GetPerformedBy();
payout.ProcessedAt = DateTime.UtcNow;
payout.LastModified = DateTime.UtcNow;
payout.ProcessedAt = DateTime.Now;
payout.LastModified = DateTime.Now;
// TODO: Add PayoutHistory record
// var history = new CommissionPayoutHistory
@@ -51,7 +51,7 @@ public class ApproveWithdrawalCommandHandler : IRequestHandler<ApproveWithdrawal
// Action = (int)CommissionPayoutAction.Approved,
// PerformedBy = "Admin", // TODO: Get from authenticated user
// Reason = request.Notes,
// Created = DateTime.UtcNow
// Created = DateTime.Now
// };
// _context.CommissionPayoutHistories.Add(history);

View File

@@ -28,9 +28,17 @@ public class CalculateWeeklyBalancesCommandHandler : IRequestHandler<CalculateWe
await _context.SaveChangesAsync(cancellationToken);
}
// دریافت کاربران فعال در شبکه
// دریافت همه کاربرانی که عضو فعال باشگاه هستند
// بدون محدودیت زمانی - همه اعضای فعال کلاب باید کمیسیون بگیرند
var activeClubMemberUserIds = await _context.ClubMemberships
.Where(c => c.IsActive)
.Select(c => c.UserId)
.ToHashSetAsync(cancellationToken);
// دریافت کاربران فعال در شبکه که عضو باشگاه هستند
// نکته: شرط NetworkParentId.HasValue نداریم چون ریشه شبکه (اولین نفر) هم باید حساب بشه
var usersInNetwork = await _context.Users
.Where(x => x.NetworkParentId.HasValue)
.Where(x => activeClubMemberUserIds.Contains(x.Id))
.Select(x => new { x.Id })
.ToListAsync(cancellationToken);
@@ -47,7 +55,7 @@ public class CalculateWeeklyBalancesCommandHandler : IRequestHandler<CalculateWe
.ToDictionaryAsync(x => x.UserId, cancellationToken);
var balancesList = new List<NetworkWeeklyBalance>();
var calculatedAt = DateTime.UtcNow;
var calculatedAt = DateTime.Now;
// خواندن یکباره Configuration ها (بهینه‌سازی - به جای N query)
var configs = await _context.SystemConfigurations
@@ -57,15 +65,15 @@ public class CalculateWeeklyBalancesCommandHandler : IRequestHandler<CalculateWe
x.Key == "Commission.MaxWeeklyBalancesPerLeg" ||
x.Key == "Commission.MaxNetworkLevel"))
.ToDictionaryAsync(x => x.Key, x => x.Value, cancellationToken);
var activationFee = long.Parse(configs.GetValueOrDefault("Club.ActivationFee", "25000000"));
var poolPercent = decimal.Parse(configs.GetValueOrDefault("Commission.WeeklyPoolContributionPercent", "20")) / 100m;
// var activationFee = long.Parse(configs.GetValueOrDefault("Club.ActivationFee", "25000000"));
// var poolPercent = decimal.Parse(configs.GetValueOrDefault("Commission.WeeklyPoolContributionPercent", "20")) / 100m;
// سقف تعادل هفتگی برای هر دست (نه کل) - 300 برای چپ + 300 برای راست = حداکثر 600 تعادل
var maxBalancesPerLeg = int.Parse(configs.GetValueOrDefault("Commission.MaxWeeklyBalancesPerLeg", "300"));
// حداکثر عمق شبکه برای شمارش اعضا (15 لول)
var maxNetworkLevel = int.Parse(configs.GetValueOrDefault("Commission.MaxNetworkLevel", "15"));
foreach (var user in usersInNetwork)
foreach (var user in usersInNetwork.OrderBy(o=>o.Id))
{
// دریافت باقیمانده هفته قبل
var leftCarryover = 0;
@@ -84,29 +92,34 @@ public class CalculateWeeklyBalancesCommandHandler : IRequestHandler<CalculateWe
var leftTotal = leftNewMembers + leftCarryover;
var rightTotal = rightNewMembers + rightCarryover;
// ✅ اصلاح شده: اعمال سقف روی هر دست جداگانه (نه روی کل)
// سقف 300 برای دست چپ + 300 برای دست راست = حداکثر 600 تعادل در هفته
var cappedLeftTotal = Math.Min(leftTotal, maxBalancesPerLeg);
var cappedRightTotal = Math.Min(rightTotal, maxBalancesPerLeg);
// ✅ مرحله 1: محاسبه تعادل اولیه (قبل از اعمال سقف)
// تعادل = کمترین مقدار بین چپ و راست
// مثال: چپ=500، راست=600 → تعادل=500
var totalBalances = Math.Min(leftTotal, rightTotal);
// محاسبه تعادل (کمترین مقدار بعد از اعمال سقف)
var totalBalances = Math.Min(cappedLeftTotal, cappedRightTotal);
// ✅ مرحله 2: محاسبه باقیمانده (قبل از سقف)
// باقیمانده = اضافه‌ای که یک طرف دارد
// مثال: چپ=500، راست=600، تعادل=500
// → باقی چپ = 500-500 = 0
// → باقی راست = 600-500 = 100 (می‌رود برای هفته بعد)
var leftRemainder = leftTotal - totalBalances;
var rightRemainder = rightTotal - totalBalances;
// محاسبه باقیمانده برای هفته بعد
// باقیمانده = مقداری که از سقف هر دست رد شده
// مثال: چپ=350، راست=450، سقف=300
// cappedLeft = MIN(350, 300) = 300
// cappedRight = MIN(450, 300) = 300
// totalBalances = MIN(300, 300) = 300
// leftRemainder = 350 - 300 = 50 (مازاد سقف)
// rightRemainder = 450 - 300 = 150 (مازاد سقف)
var leftRemainder = leftTotal - cappedLeftTotal;
var rightRemainder = rightTotal - cappedRightTotal;
// ✅ مرحله 3: اعمال سقف 300 (برای امتیاز نهایی)
// از تعادل، فقط 300 از هر طرف حساب می‌شود
// مثال: تعادل=500 → امتیاز=300
// از چپ: 300 حساب می‌شود، 200 فلش می‌شود
// از راست: 300 حساب می‌شود، 200 فلش می‌شود
// جمع فلش = 400 (از بین می‌رود)
var cappedBalances = Math.Min(totalBalances, maxBalancesPerLeg);
// ✅ مرحله 4: محاسبه فلش (از هر دو طرف)
var flushedPerSide = totalBalances - cappedBalances; // 500-300=200
var totalFlushed = flushedPerSide * 2; // 200×2=400 (از بین می‌رود)
// محاسبه سهم استخر (20% از مجموع فعال‌سازی‌های جدید کل شبکه)
// طبق گفته دکتر: کل افراد جدید در شبکه × هزینه فعال‌سازی × 20%
var totalNewMembers = leftNewMembers + rightNewMembers;
var weeklyPoolContribution = (long)(totalNewMembers * activationFee * poolPercent);
// ⚠️ توجه: تعادل زیرمجموعه در این مرحله محاسبه نمیشه
// چون هنوز تمام تعادل‌ها محاسبه نشدن
// بعد از ذخیره همه تعادل‌ها، در یک حلقه دوم محاسبه خواهد شد
var balance = new NetworkWeeklyBalance
{
@@ -122,19 +135,26 @@ public class CalculateWeeklyBalancesCommandHandler : IRequestHandler<CalculateWe
// مجموع
LeftLegTotal = leftTotal,
RightLegTotal = rightTotal,
TotalBalances = totalBalances, // تعادل واقعی بعد از اعمال سقف روی هر دست
TotalBalances = cappedBalances, // امتیاز نهایی بعد از اعمال سقف 300
// باقیمانده برای هفته بعد (مازاد سقف هر دست)
// باقیمانده برای هفته بعد (اضافه‌ای که یک طرف دارد)
LeftLegRemainder = leftRemainder,
RightLegRemainder = rightRemainder,
// فلش (از دست رفته)
FlushedPerSide = flushedPerSide,
TotalFlushed = totalFlushed,
// تعادل زیرمجموعه - فعلاً 0 (بعد از ذخیره همه تعادل‌ها محاسبه میشه)
SubordinateBalances = 0,
// فیلدهای قدیمی (deprecated) - برای سازگاری با کدهای قبلی
#pragma warning disable CS0618
LeftLegBalances = leftTotal,
RightLegBalances = rightTotal,
#pragma warning restore CS0618
WeeklyPoolContribution = weeklyPoolContribution,
WeeklyPoolContribution = 0, // Pool در مرحله بعد محاسبه میشه
CalculatedAt = calculatedAt,
IsExpired = false
};
@@ -145,6 +165,26 @@ public class CalculateWeeklyBalancesCommandHandler : IRequestHandler<CalculateWe
await _context.NetworkWeeklyBalances.AddRangeAsync(balancesList, cancellationToken);
await _context.SaveChangesAsync(cancellationToken);
// ⭐ مرحله 2: محاسبه تعادل زیرمجموعه برای هر کاربر (تا 15 لول)
// حالا که همه تعادل‌ها ذخیره شدن، می‌تونیم تعادل زیرمجموعه رو حساب کنیم
var balancesDictionary = balancesList.ToDictionary(x => x.UserId);
foreach (var balance in balancesList)
{
var subordinateBalances = await CalculateSubordinateBalancesAsync(
balance.UserId,
balancesDictionary,
maxNetworkLevel,
cancellationToken
);
balance.SubordinateBalances = subordinateBalances;
}
// ذخیره تعادل‌های زیرمجموعه
_context.NetworkWeeklyBalances.UpdateRange(balancesList);
await _context.SaveChangesAsync(cancellationToken);
return balancesList.Count;
}
@@ -239,13 +279,76 @@ public class CalculateWeeklyBalancesCommandHandler : IRequestHandler<CalculateWe
var year = int.Parse(parts[0]);
var week = int.Parse(parts[1].Replace("W", ""));
// محاسبه اولین روز هفته (شنبه)
// محاسبه اولین شنبه سال
var jan1 = new DateTime(year, 1, 1);
var daysOffset = DayOfWeek.Saturday - jan1.DayOfWeek;
var firstSaturday = jan1.AddDays(daysOffset);
var jan1DayOfWeek = (int)jan1.DayOfWeek;
// اگر 1 ژانویه شنبه باشد: offset=0، اگر یکشنبه: offset=6، دوشنبه: offset=5، ...
var daysToFirstSaturday = jan1DayOfWeek == 6 ? 0 : (6 - jan1DayOfWeek + 7) % 7;
var firstSaturday = jan1.AddDays(daysToFirstSaturday);
var weekStart = firstSaturday.AddDays((week - 1) * 7);
var weekEnd = weekStart.AddDays(6).AddHours(23).AddMinutes(59).AddSeconds(59);
return (weekStart, weekEnd);
}
/// <summary>
/// محاسبه مجموع تعادل‌های زیرمجموعه یک کاربر تا maxLevel لول پایین‌تر
/// </summary>
private async Task<int> CalculateSubordinateBalancesAsync(
long userId,
Dictionary<long, NetworkWeeklyBalance> allBalances,
int maxLevel,
CancellationToken cancellationToken)
{
// پیدا کردن همه زیرمجموعه‌ها تا maxLevel لول
var subordinates = await GetSubordinatesRecursive(userId, 1, maxLevel, cancellationToken);
// جمع تعادل‌های آنها
var totalSubordinateBalances = 0;
foreach (var subordinateId in subordinates)
{
if (allBalances.ContainsKey(subordinateId))
{
totalSubordinateBalances += allBalances[subordinateId].TotalBalances;
}
}
return totalSubordinateBalances;
}
/// <summary>
/// پیدا کردن بازگشتی زیرمجموعه‌ها تا maxLevel لول
/// </summary>
private async Task<List<long>> GetSubordinatesRecursive(
long userId,
int currentLevel,
int maxLevel,
CancellationToken cancellationToken)
{
// محدودیت عمق
if (currentLevel > maxLevel)
{
return new List<long>();
}
var result = new List<long>();
// پیدا کردن فرزندان مستقیم
var children = await _context.Users
.Where(x => x.NetworkParentId == userId)
.Select(x => x.Id)
.ToListAsync(cancellationToken);
result.AddRange(children);
// بازگشت برای هر فرزند
foreach (var childId in children)
{
var grandChildren = await GetSubordinatesRecursive(childId, currentLevel + 1, maxLevel, cancellationToken);
result.AddRange(grandChildren);
}
return result;
}
}

View File

@@ -11,11 +11,19 @@ public class CalculateWeeklyCommissionPoolCommandHandler : IRequestHandler<Calcu
public async Task<long> Handle(CalculateWeeklyCommissionPoolCommand request, CancellationToken cancellationToken)
{
// بررسی وجود استخر قبلی
// بررسی وجود استخر
var existingPool = await _context.WeeklyCommissionPools
.FirstOrDefaultAsync(x => x.WeekNumber == request.WeekNumber, cancellationToken);
if (existingPool != null && existingPool.IsCalculated && !request.ForceRecalculate)
if (existingPool == null)
{
throw new InvalidOperationException(
$"Pool هفته {request.WeekNumber} وجود ندارد. " +
"Pool باید در هنگام فعالسازی باشگاه مشتریان ایجاد شده باشد"
);
}
if (existingPool.IsCalculated && !request.ForceRecalculate)
{
throw new InvalidOperationException($"استخر کمیسیون هفته {request.WeekNumber} قبلاً محاسبه شده است");
}
@@ -30,48 +38,122 @@ public class CalculateWeeklyCommissionPoolCommandHandler : IRequestHandler<Calcu
throw new InvalidOperationException($"تعادل‌های هفته {request.WeekNumber} هنوز محاسبه نشده است. ابتدا CalculateWeeklyBalances را اجرا کنید");
}
// محاسبه مجموع مشارکت‌ها در استخر
var totalPoolAmount = weeklyBalances.Sum(x => x.WeeklyPoolContribution);
// ⭐ Pool از قبل پُر شده (توسط ActivateClubMembership)
var totalPoolAmount = existingPool.TotalPoolAmount;
// محاسبه مجموع Balances
var totalBalances = weeklyBalances.Sum(x => x.TotalBalances);
// محاسبه مجموع تعادل‌های کل شبکه
// نکته: SubordinateBalances اضافه نمی‌کنیم چون وقتی همه TotalBalances رو جمع می‌زنیم،
// خودش شامل بالانس‌های زیرمجموعه‌ها هم هست (تکراری نشه)
var totalBalancesInNetwork = weeklyBalances.Sum(x => x.TotalBalances);
// محاسبه ارزش هر Balance (تقسیم صحیح برای ریال)
// محاسبه ارزش هر امتیاز
long valuePerBalance = 0;
if (totalBalances > 0)
if (totalBalancesInNetwork > 0)
{
valuePerBalance = totalPoolAmount / totalBalances;
valuePerBalance = totalPoolAmount / totalBalancesInNetwork;
}
if (existingPool != null)
{
// به‌روزرسانی
existingPool.TotalPoolAmount = totalPoolAmount;
existingPool.TotalBalances = totalBalances;
existingPool.ValuePerBalance = valuePerBalance;
existingPool.IsCalculated = true;
existingPool.CalculatedAt = DateTime.UtcNow;
// به‌روزرسانی Pool
existingPool.TotalBalances = totalBalancesInNetwork;
existingPool.ValuePerBalance = valuePerBalance;
existingPool.IsCalculated = true;
existingPool.CalculatedAt = DateTime.Now;
_context.WeeklyCommissionPools.Update(existingPool);
}
else
_context.WeeklyCommissionPools.Update(existingPool);
await _context.SaveChangesAsync(cancellationToken);
// حذف پرداخت‌های قبلی در صورت ForceRecalculate
if (request.ForceRecalculate)
{
// ایجاد جدید
var pool = new WeeklyCommissionPool
var oldPayouts = await _context.UserCommissionPayouts
.Where(p => p.WeekNumber == request.WeekNumber)
.ToListAsync(cancellationToken);
if (oldPayouts.Any())
{
var oldPayoutIds = oldPayouts.Select(p => p.Id).ToList();
// ⭐ اول باید تاریخچه‌ها حذف بشن (به خاطر FK constraint)
var oldHistories = await _context.CommissionPayoutHistories
.Where(h => oldPayoutIds.Contains(h.UserCommissionPayoutId))
.ToListAsync(cancellationToken);
if (oldHistories.Any())
{
_context.CommissionPayoutHistories.RemoveRange(oldHistories);
}
// بعد پرداخت‌ها حذف می‌شن
_context.UserCommissionPayouts.RemoveRange(oldPayouts);
await _context.SaveChangesAsync(cancellationToken);
}
}
// ⭐ ثبت پرداخت برای کاربرانی که تعادل دارند
// تعادل شخصی + زیرمجموعه قبلاً در CalculateWeeklyBalances محاسبه شده
var payouts = new List<UserCommissionPayout>();
foreach (var balance in weeklyBalances)
{
// فقط تعادل شخصی (SubordinateBalances اضافه نمی‌شه چون در SUM کل شبکه خودش حساب میشه)
var userBalance = balance.TotalBalances;
// اگر تعادل صفر است، نیازی به ثبت نیست
if (userBalance <= 0)
{
continue;
}
// محاسبه مبلغ کمیسیون
var totalAmount = (long)(userBalance * valuePerBalance);
var payout = new UserCommissionPayout
{
UserId = balance.UserId,
WeekNumber = request.WeekNumber,
TotalPoolAmount = totalPoolAmount,
TotalBalances = totalBalances,
WeeklyPoolId = existingPool.Id,
BalancesEarned = userBalance,
ValuePerBalance = valuePerBalance,
IsCalculated = true,
CalculatedAt = DateTime.UtcNow
TotalAmount = totalAmount,
Status = CommissionPayoutStatus.Pending,
PaidAt = null,
WithdrawalMethod = null,
IbanNumber = null,
WithdrawnAt = null
};
await _context.WeeklyCommissionPools.AddAsync(pool, cancellationToken);
existingPool = pool;
payouts.Add(payout);
}
await _context.SaveChangesAsync(cancellationToken);
if (payouts.Any())
{
await _context.UserCommissionPayouts.AddRangeAsync(payouts, cancellationToken);
await _context.SaveChangesAsync(cancellationToken);
// ثبت تاریخچه برای هر پرداخت
var historyList = new List<CommissionPayoutHistory>();
foreach (var payout in payouts)
{
var history = new CommissionPayoutHistory
{
UserCommissionPayoutId = payout.Id,
UserId = payout.UserId,
WeekNumber = request.WeekNumber,
AmountBefore = 0,
AmountAfter = payout.TotalAmount,
OldStatus = default(CommissionPayoutStatus),
NewStatus = CommissionPayoutStatus.Pending,
Action = CommissionPayoutAction.Created,
PerformedBy = "System",
Reason = "پردازش خودکار کمیسیون هفتگی"
};
historyList.Add(history);
}
await _context.CommissionPayoutHistories.AddRangeAsync(historyList, cancellationToken);
await _context.SaveChangesAsync(cancellationToken);
}
return existingPool.Id;
}

View File

@@ -39,7 +39,7 @@ public class ProcessWithdrawalCommandHandler : IRequestHandler<ProcessWithdrawal
}
var oldStatus = payout.Status;
var now = DateTime.UtcNow;
var now = DateTime.Now;
if (request.IsApproved)
{

View File

@@ -34,9 +34,9 @@ public class RejectWithdrawalCommandHandler : IRequestHandler<RejectWithdrawalCo
// Update status to Cancelled (rejected)
payout.Status = CommissionPayoutStatus.Cancelled;
payout.ProcessedBy = _currentUser.GetPerformedBy();
payout.ProcessedAt = DateTime.UtcNow;
payout.ProcessedAt = DateTime.Now;
payout.RejectionReason = request.Reason;
payout.LastModified = DateTime.UtcNow;
payout.LastModified = DateTime.Now;
// TODO: Add PayoutHistory record with rejection reason
// var history = new CommissionPayoutHistory
@@ -51,7 +51,7 @@ public class RejectWithdrawalCommandHandler : IRequestHandler<RejectWithdrawalCo
// Action = (int)CommissionPayoutAction.Rejected,
// PerformedBy = "Admin", // TODO: Get from authenticated user
// Reason = request.Reason,
// Created = DateTime.UtcNow
// Created = DateTime.Now
// };
// _context.CommissionPayoutHistories.Add(history);

View File

@@ -13,17 +13,12 @@ public record TriggerWeeklyCalculationCommand : IRequest<TriggerWeeklyCalculatio
public bool ForceRecalculate { get; init; }
/// <summary>
/// Skip balance calculation
/// Skip balance calculation (Step 1)
/// </summary>
public bool SkipBalances { get; init; }
/// <summary>
/// Skip pool calculation
/// </summary>
public bool SkipPool { get; init; }
/// <summary>
/// Skip payout processing
/// Skip pool calculation and payout processing (Step 2)
/// </summary>
public bool SkipPayouts { get; init; }
}

View File

@@ -1,7 +1,6 @@
using CMSMicroservice.Application.Common.Interfaces;
using CMSMicroservice.Application.CommissionCQ.Commands.CalculateWeeklyBalances;
using CMSMicroservice.Application.CommissionCQ.Commands.CalculateWeeklyCommissionPool;
using CMSMicroservice.Application.CommissionCQ.Commands.ProcessUserPayouts;
namespace CMSMicroservice.Application.CommissionCQ.Commands.TriggerWeeklyCalculation;
@@ -23,7 +22,7 @@ public class TriggerWeeklyCalculationCommandHandler : IRequestHandler<TriggerWee
CancellationToken cancellationToken)
{
var executionId = Guid.NewGuid().ToString();
var startedAt = DateTime.UtcNow;
var startedAt = DateTime.Now;
try
{
@@ -41,7 +40,7 @@ public class TriggerWeeklyCalculationCommandHandler : IRequestHandler<TriggerWee
var steps = new List<string>();
// Step 1: Calculate Weekly Balances
// Step 1: Calculate Weekly Balances (تا 15 لول)
if (!request.SkipBalances)
{
await _mediator.Send(new CalculateWeeklyBalancesCommand
@@ -52,25 +51,15 @@ public class TriggerWeeklyCalculationCommandHandler : IRequestHandler<TriggerWee
steps.Add("محاسبه امتیازات هفتگی");
}
// Step 2: Calculate Weekly Commission Pool
if (!request.SkipPool)
// Step 2: Calculate Pool & Process Payouts (محاسبه استخر + پرداخت کاربران)
if (!request.SkipPayouts)
{
await _mediator.Send(new CalculateWeeklyCommissionPoolCommand
{
WeekNumber = request.WeekNumber
}, cancellationToken);
steps.Add("محاسبه استخر کمیسیون");
}
// Step 3: Process User Payouts
if (!request.SkipPayouts)
{
await _mediator.Send(new ProcessUserPayoutsCommand
{
WeekNumber = request.WeekNumber,
ForceReprocess = request.ForceRecalculate
ForceRecalculate = request.ForceRecalculate
}, cancellationToken);
steps.Add("پردازش پرداخت‌های کاربران");
steps.Add("محاسبه استخر و پرداخت کاربران");
}
return new TriggerWeeklyCalculationResponseDto

View File

@@ -0,0 +1,13 @@
namespace CMSMicroservice.Application.CommissionCQ.Queries.GetAvailableWeeks;
/// <summary>
/// دریافت لیست هفته‌های قابل انتخاب برای محاسبه کمیسیون
/// </summary>
public class GetAvailableWeeksQuery : IRequest<GetAvailableWeeksResponseDto>
{
/// <summary>تعداد هفته‌های آینده برای نمایش (پیش‌فرض: 4)</summary>
public int FutureWeeksCount { get; init; } = 4;
/// <summary>تعداد هفته‌های گذشته برای نمایش (پیش‌فرض: 12)</summary>
public int PastWeeksCount { get; init; } = 12;
}

View File

@@ -0,0 +1,192 @@
using CMSMicroservice.Application.Common.Interfaces;
using CMSMicroservice.Domain.Enums;
using Microsoft.EntityFrameworkCore;
using System.Globalization;
namespace CMSMicroservice.Application.CommissionCQ.Queries.GetAvailableWeeks;
public class GetAvailableWeeksQueryHandler : IRequestHandler<GetAvailableWeeksQuery, GetAvailableWeeksResponseDto>
{
private readonly IApplicationDbContext _context;
public GetAvailableWeeksQueryHandler(IApplicationDbContext context)
{
_context = context;
}
public async Task<GetAvailableWeeksResponseDto> Handle(
GetAvailableWeeksQuery request,
CancellationToken cancellationToken)
{
var currentDate = DateTime.Now;
var currentWeekNumber = GetWeekNumber(currentDate);
// دریافت هفته‌های محاسبه شده از دیتابیس
var calculatedPools = await _context.WeeklyCommissionPools
.Where(p => p.IsCalculated)
.OrderByDescending(p => p.WeekNumber)
.Take(request.PastWeeksCount)
.ToListAsync(cancellationToken);
// دریافت لاگ‌های اجرا
var executionLogs = await _context.WorkerExecutionLogs
.Where(log => log.Status == WorkerExecutionStatus.Success ||
log.Status == WorkerExecutionStatus.Failed)
.GroupBy(log => log.WeekNumber)
.Select(g => new
{
WeekNumber = g.Key,
LastLog = g.OrderByDescending(l => l.StartedAt).First()
})
.ToDictionaryAsync(x => x.WeekNumber, x => x.LastLog, cancellationToken);
var allWeeks = new List<WeekInfoDto>();
// هفته جاری
var currentWeekInfo = CreateWeekInfo(currentDate, currentWeekNumber, calculatedPools, executionLogs);
// هفته‌های گذشته (12 هفته)
var pastWeeks = new List<WeekInfoDto>();
for (int i = 1; i <= request.PastWeeksCount; i++)
{
var pastDate = currentDate.AddDays(-7 * i);
var weekNumber = GetWeekNumber(pastDate);
pastWeeks.Add(CreateWeekInfo(pastDate, weekNumber, calculatedPools, executionLogs));
}
// هفته‌های آینده (4 هفته)
var futureWeeks = new List<WeekInfoDto>();
for (int i = 1; i <= request.FutureWeeksCount; i++)
{
var futureDate = currentDate.AddDays(7 * i);
var weekNumber = GetWeekNumber(futureDate);
futureWeeks.Add(CreateWeekInfo(futureDate, weekNumber, calculatedPools, executionLogs));
}
// تفکیک به calculated و pending
var calculatedWeeks = pastWeeks.Where(w => w.IsCalculated).ToList();
var pendingWeeks = pastWeeks.Where(w => !w.IsCalculated).ToList();
return new GetAvailableWeeksResponseDto
{
CurrentWeek = currentWeekInfo,
CalculatedWeeks = calculatedWeeks,
PendingWeeks = pendingWeeks,
FutureWeeks = futureWeeks
};
}
private WeekInfoDto CreateWeekInfo(
DateTime date,
string weekNumber,
List<Domain.Entities.Commission.WeeklyCommissionPool> calculatedPools,
Dictionary<string, Domain.Entities.Commission.WorkerExecutionLog> executionLogs)
{
var (startDate, endDate) = GetWeekRange(date);
var pool = calculatedPools.FirstOrDefault(p => p.WeekNumber == weekNumber);
var log = executionLogs.GetValueOrDefault(weekNumber);
var isCalculated = pool != null && pool.IsCalculated;
// تبدیل تاریخ به شمسی برای نمایش
var persianCalendar = new PersianCalendar();
var startDatePersian = $"{persianCalendar.GetYear(startDate):D4}/{persianCalendar.GetMonth(startDate):D2}/{persianCalendar.GetDayOfMonth(startDate):D2}";
var endDatePersian = $"{persianCalendar.GetYear(endDate):D4}/{persianCalendar.GetMonth(endDate):D2}/{persianCalendar.GetDayOfMonth(endDate):D2}";
// تبدیل weekNumber به شمسی فقط برای نمایش
var persianWeekNumber = ConvertWeekNumberToPersian(weekNumber, startDate);
var displayText = $"{persianWeekNumber} ({startDatePersian} - {endDatePersian})";
if (isCalculated)
{
displayText += " ✅ محاسبه شده";
}
return new WeekInfoDto
{
WeekNumber = weekNumber, // میلادی (برای API)
StartDate = startDate,
EndDate = endDate,
IsCalculated = isCalculated,
CalculatedAt = pool?.CalculatedAt,
LastExecutionStatus = log?.Status.ToString(),
TotalPoolAmount = pool?.TotalPoolAmount,
EligibleUsersCount = pool?.UserCommissionPayouts?.Count ?? 0,
DisplayText = displayText // شمسی (برای نمایش)
};
}
private static string GetWeekNumber(DateTime date)
{
var year = date.Year;
// پیدا کردن اولین شنبه سال
var jan1 = new DateTime(year, 1, 1);
var jan1DayOfWeek = (int)jan1.DayOfWeek; // 0=Sunday, 1=Monday, ..., 6=Saturday
// محاسبه تعداد روزهایی که باید اضافه کنیم تا به اولین شنبه برسیم
var daysToFirstSaturday = jan1DayOfWeek == 6 ? 0 : (6 - jan1DayOfWeek + 7) % 7;
var firstSaturday = jan1.AddDays(daysToFirstSaturday);
// پیدا کردن شنبه شروع هفته جاری
var currentDayOfWeek = (int)date.DayOfWeek; // 0=Sun, 1=Mon, ..., 6=Sat
var daysToCurrentSaturday = currentDayOfWeek == 6 ? 0 : (currentDayOfWeek + 1) % 7;
var weekStartSaturday = date.Date.AddDays(-daysToCurrentSaturday);
// محاسبه شماره هفته
int weekNum;
if (weekStartSaturday < firstSaturday)
{
weekNum = 1; // هفته اول سال
}
else
{
var daysSinceFirstSaturday = (weekStartSaturday - firstSaturday).Days;
weekNum = (daysSinceFirstSaturday / 7) + 1;
}
return $"{year}-W{weekNum:D2}";
}
private static (DateTime startDate, DateTime endDate) GetWeekRange(DateTime date)
{
var dayOfWeek = (int)date.DayOfWeek;
// محاسبه تعداد روزهایی که باید عقب برویم تا به شنبه برسیم
// شنبه = 6, یکشنبه = 0, دوشنبه = 1, ..., جمعه = 5
var daysToSaturday = dayOfWeek == 6 ? 0 : (dayOfWeek + 1) % 7;
var startDate = date.Date.AddDays(-daysToSaturday);
var endDate = startDate.AddDays(6).AddHours(23).AddMinutes(59).AddSeconds(59);
return (startDate, endDate);
}
private static string ConvertWeekNumberToPersian(string gregorianWeekNumber, DateTime weekStartDate)
{
// ورودی: "2025-W48"
var persianCalendar = new PersianCalendar();
var persianYear = persianCalendar.GetYear(weekStartDate);
// محاسبه شماره هفته شمسی
var jan1Persian = persianCalendar.ToDateTime(persianYear, 1, 1, 0, 0, 0, 0);
// پیدا کردن اولین شنبه سال شمسی
var jan1DayOfWeek = (int)jan1Persian.DayOfWeek;
var daysToFirstSaturday = jan1DayOfWeek == 6 ? 0 : (6 - jan1DayOfWeek + 7) % 7;
var firstSaturday = jan1Persian.AddDays(daysToFirstSaturday);
// محاسبه شماره هفته
int weekNum;
if (weekStartDate < firstSaturday)
{
weekNum = 1;
}
else
{
var daysSinceFirstSaturday = (weekStartDate - firstSaturday).Days;
weekNum = (daysSinceFirstSaturday / 7) + 1;
}
return $"{persianYear}-W{weekNum:D2}";
}
}

View File

@@ -0,0 +1,46 @@
namespace CMSMicroservice.Application.CommissionCQ.Queries.GetAvailableWeeks;
public class GetAvailableWeeksResponseDto
{
/// <summary>هفته جاری</summary>
public required WeekInfoDto CurrentWeek { get; init; }
/// <summary>هفته‌های محاسبه شده (از جدیدترین به قدیمی‌ترین)</summary>
public required List<WeekInfoDto> CalculatedWeeks { get; init; }
/// <summary>هفته‌های محاسبه نشده (از جدیدترین به قدیمی‌ترین)</summary>
public required List<WeekInfoDto> PendingWeeks { get; init; }
/// <summary>هفته‌های آینده قابل انتخاب</summary>
public required List<WeekInfoDto> FutureWeeks { get; init; }
}
public class WeekInfoDto
{
/// <summary>شماره هفته (YYYY-Www)</summary>
public required string WeekNumber { get; init; }
/// <summary>تاریخ شروع هفته</summary>
public required DateTime StartDate { get; init; }
/// <summary>تاریخ پایان هفته</summary>
public required DateTime EndDate { get; init; }
/// <summary>آیا محاسبه شده؟</summary>
public bool IsCalculated { get; init; }
/// <summary>تاریخ محاسبه (اگر محاسبه شده باشد)</summary>
public DateTime? CalculatedAt { get; init; }
/// <summary>وضعیت اجرای آخرین محاسبه</summary>
public string? LastExecutionStatus { get; init; }
/// <summary>مبلغ کل استخر کمیسیون (اگر محاسبه شده باشد)</summary>
public long? TotalPoolAmount { get; init; }
/// <summary>تعداد کاربران واجد شرایط</summary>
public int? EligibleUsersCount { get; init; }
/// <summary>نمایش فارسی (برای UI)</summary>
public required string DisplayText { get; init; }
}

View File

@@ -31,7 +31,7 @@ public class GetCommissionPayoutHistoryQueryHandler : IRequestHandler<GetCommiss
query = query.Where(x => x.WeekNumber == request.WeekNumber);
}
query = query.ApplyOrder(sortBy: request.SortBy ?? "-Created");
query = query.ApplyOrder(sortBy: request.SortBy ?? "Created");
var meta = await query.GetMetaData(request.PaginationState, cancellationToken);

View File

@@ -31,7 +31,7 @@ public class GetUserCommissionPayoutsQueryHandler : IRequestHandler<GetUserCommi
query = query.Where(x => x.WeekNumber == request.WeekNumber);
}
query = query.ApplyOrder(sortBy: request.SortBy ?? "-Created");
query = query.ApplyOrder(sortBy: request.SortBy ?? "Created");
var meta = await query.GetMetaData(request.PaginationState, cancellationToken);

View File

@@ -31,7 +31,9 @@ public class GetUserWeeklyBalancesQueryHandler : IRequestHandler<GetUserWeeklyBa
query = query.Where(x => !x.IsExpired);
}
query = query.ApplyOrder(sortBy: request.SortBy ?? "-WeekNumber");
// نمی‌توانیم "-WeekNumber" استفاده کنیم چون WeekNumber یک string است
// از Created برای مرتب‌سازی استفاده می‌کنیم (جدیدترین هفته‌ها اول)
query = query.ApplyOrder(sortBy: request.SortBy ?? "Created");
var meta = await query.GetMetaData(request.PaginationState, cancellationToken);

View File

@@ -27,6 +27,6 @@ public class GetWeeklyCommissionPoolQueryHandler : IRequestHandler<GetWeeklyComm
})
.FirstOrDefaultAsync(cancellationToken);
return pool;
return pool??new WeeklyCommissionPoolDto();
}
}

View File

@@ -19,7 +19,7 @@ public class GetWithdrawalReportsQueryHandler : IRequestHandler<GetWithdrawalRep
public async Task<WithdrawalReportsDto> Handle(GetWithdrawalReportsQuery request, CancellationToken cancellationToken)
{
// تعیین بازه زمانی پیش‌فرض (30 روز گذشته)
var endDate = request.EndDate ?? DateTime.UtcNow;
var endDate = request.EndDate ?? DateTime.Now;
var startDate = request.StartDate ?? endDate.AddDays(-30);
// Query پایه

View File

@@ -38,7 +38,7 @@ public class GetWithdrawalRequestsQueryHandler : IRequestHandler<GetWithdrawalRe
query = query.Where(x => x.IbanNumber != null && x.IbanNumber.Contains(request.IbanNumber));
}
query = query.ApplyOrder(sortBy: request.SortBy ?? "-Created");
query = query.ApplyOrder(sortBy: request.SortBy ?? "Created");
var meta = await query.GetMetaData(request.PaginationState, cancellationToken);

View File

@@ -27,8 +27,8 @@ public class GetWorkerStatusQueryHandler : IRequestHandler<GetWorkerStatusQuery,
CurrentExecutionId = null,
CurrentWeekNumber = null,
CurrentStep = "Idle",
LastRunAt = DateTime.UtcNow.AddHours(-24),
NextScheduledRun = DateTime.UtcNow.AddDays(7),
LastRunAt = DateTime.Now.AddHours(-24),
NextScheduledRun = DateTime.Now.AddDays(7),
TotalExecutions = 48,
SuccessfulExecutions = 47,
FailedExecutions = 1

View File

@@ -22,7 +22,7 @@ public class GetConfigurationHistoryQueryHandler : IRequestHandler<GetConfigurat
var query = _context.SystemConfigurationHistories
.Where(x => x.ConfigurationId == request.ConfigurationId)
.ApplyOrder(sortBy: request.SortBy ?? "-Created") // پیش‌فرض: جدیدترین اول
.ApplyOrder(sortBy: request.SortBy ?? "Created") // پیش‌فرض: جدیدترین اول
.AsNoTracking()
.AsQueryable();

View File

@@ -48,7 +48,7 @@ public class CheckDayaLoanStatusCommandHandler : IRequestHandler<CheckDayaLoanSt
if (existingContract != null)
{
existingContract.LastCheckDate = DateTime.UtcNow;
existingContract.LastCheckDate = DateTime.Now;
existingContract.Status = dayaResult.Status;
existingContract.ContractNumber = dayaResult.ContractNumber;
}
@@ -65,7 +65,7 @@ public class CheckDayaLoanStatusCommandHandler : IRequestHandler<CheckDayaLoanSt
NationalCode = dayaResult.NationalCode,
Status = dayaResult.Status,
ContractNumber = dayaResult.ContractNumber,
LastCheckDate = DateTime.UtcNow,
LastCheckDate = DateTime.Now,
IsProcessed = false
};

View File

@@ -36,7 +36,7 @@ public class ProcessDayaLoanApprovalCommandHandler : IRequestHandler<ProcessDaya
Amount = request.WalletAmount + request.LockedWalletAmount + request.DiscountWalletAmount, // 168 میلیون
Description = $"دریافت اعتبار دایا - قرارداد {request.ContractNumber}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
RefId = request.ContractNumber, // شماره قرارداد دایا
Type = TransactionType.DepositExternal1
};
@@ -114,7 +114,7 @@ public class ProcessDayaLoanApprovalCommandHandler : IRequestHandler<ProcessDaya
// به‌روزرسانی وضعیت کاربر
user.HasReceivedDayaCredit = true;
user.DayaCreditReceivedAt = DateTime.UtcNow;
user.DayaCreditReceivedAt = DateTime.Now;
// تنظیم نحوه خرید پکیج به DayaLoan
user.PackagePurchaseMethod = PackagePurchaseMethod.DayaLoan;
@@ -139,7 +139,7 @@ public class ProcessDayaLoanApprovalCommandHandler : IRequestHandler<ProcessDaya
PackageId = goldenPackage.Id,
Amount = request.WalletAmount, // 56 میلیون
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
DeliveryStatus = DeliveryStatus.None,
UserAddressId = defaultAddress.Id,
TransactionId = transaction.Id,

View File

@@ -46,12 +46,12 @@ public class CompleteOrderPaymentCommandHandler : IRequestHandler<CompleteOrderP
{
// Update transaction
transaction.PaymentStatus = PaymentStatus.Success;
transaction.PaymentDate = DateTime.UtcNow;
transaction.PaymentDate = DateTime.Now;
transaction.RefId = request.RefId;
// Update order
order.PaymentStatus = PaymentStatus.Success;
order.PaymentDate = DateTime.UtcNow;
order.PaymentDate = DateTime.Now;
order.DeliveryStatus = DeliveryStatus.InTransit;
// Deduct discount balance from user wallet

View File

@@ -47,7 +47,7 @@ public class CreateDiscountCategoryCommandHandler : IRequestHandler<CreateDiscou
ParentCategoryId = request.ParentCategoryId,
SortOrder = request.SortOrder,
IsActive = request.IsActive,
Created = DateTime.UtcNow
Created = DateTime.Now
};
_context.DiscountCategories.Add(category);

View File

@@ -86,7 +86,7 @@ public class ApproveManualPaymentCommandHandler : IRequestHandler<ApproveManualP
Amount = manualPayment.Amount,
Description = $"پرداخت دستی - {manualPayment.Type} - {manualPayment.Description}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
RefId = manualPayment.ReferenceNumber,
Type = MapToTransactionType(manualPayment.Type)
};
@@ -219,7 +219,7 @@ public class ApproveManualPaymentCommandHandler : IRequestHandler<ApproveManualP
// 7. به‌روزرسانی ManualPayment
manualPayment.Status = ManualPaymentStatus.Approved;
manualPayment.ApprovedBy = approvedById;
manualPayment.ApprovedAt = DateTime.UtcNow;
manualPayment.ApprovedAt = DateTime.Now;
manualPayment.TransactionId = transaction.Id;
await _context.SaveChangesAsync(cancellationToken);

View File

@@ -0,0 +1,24 @@
namespace CMSMicroservice.Application.ManualPaymentCQ.Commands.ProcessManualMembershipPayment;
public record ProcessManualMembershipPaymentCommand : IRequest<ProcessManualMembershipPaymentResponseDto>
{
/// <summary>
/// شناسه کاربر
/// </summary>
public long UserId { get; init; }
/// <summary>
/// مبلغ پرداختی (ریال)
/// </summary>
public long Amount { get; init; }
/// <summary>
/// شماره مرجع تراکنش
/// </summary>
public string ReferenceNumber { get; init; } = string.Empty;
/// <summary>
/// توضیحات (اختیاری)
/// </summary>
public string? Description { get; init; }
}

View File

@@ -0,0 +1,185 @@
using CMSMicroservice.Application.Common.Exceptions;
using CMSMicroservice.Application.Common.Interfaces;
using CMSMicroservice.Domain.Entities;
using CMSMicroservice.Domain.Entities.Payment;
using CMSMicroservice.Domain.Enums;
using Microsoft.EntityFrameworkCore;
namespace CMSMicroservice.Application.ManualPaymentCQ.Commands.ProcessManualMembershipPayment;
public class ProcessManualMembershipPaymentCommandHandler : IRequestHandler<ProcessManualMembershipPaymentCommand, ProcessManualMembershipPaymentResponseDto>
{
private readonly IApplicationDbContext _context;
private readonly ICurrentUserService _currentUser;
private readonly ILogger<ProcessManualMembershipPaymentCommandHandler> _logger;
public ProcessManualMembershipPaymentCommandHandler(
IApplicationDbContext context,
ICurrentUserService currentUser,
ILogger<ProcessManualMembershipPaymentCommandHandler> logger)
{
_context = context;
_currentUser = currentUser;
_logger = logger;
}
public async Task<ProcessManualMembershipPaymentResponseDto> Handle(
ProcessManualMembershipPaymentCommand request,
CancellationToken cancellationToken)
{
try
{
_logger.LogInformation(
"Processing manual membership payment for UserId: {UserId}, Amount: {Amount}",
request.UserId, request.Amount);
// 1. دریافت شناسه ادمین از کاربر جاری
var currentUserId = _currentUser.UserId;
if (string.IsNullOrEmpty(currentUserId))
{
throw new UnauthorizedAccessException("کاربر احراز هویت نشده است");
}
if (!long.TryParse(currentUserId, out var adminUserId))
{
throw new UnauthorizedAccessException("شناسه کاربر نامعتبر است");
}
// 2. بررسی وجود کاربر
var user = await _context.Users
.FirstOrDefaultAsync(u => u.Id == request.UserId, cancellationToken);
if (user == null)
{
_logger.LogError("User not found: {UserId}", request.UserId);
throw new NotFoundException($"کاربر با شناسه {request.UserId} یافت نشد");
}
// 3. ایجاد ManualPayment با وضعیت Approved
var manualPayment = new ManualPayment
{
UserId = request.UserId,
Amount = request.Amount,
Type = ManualPaymentType.CashDeposit,
Description = request.Description ?? "پرداخت دستی عضویت",
ReferenceNumber = request.ReferenceNumber,
Status = ManualPaymentStatus.Approved,
RequestedBy = adminUserId,
ApprovedBy = adminUserId,
ApprovedAt = DateTime.Now
};
_context.ManualPayments.Add(manualPayment);
await _context.SaveChangesAsync(cancellationToken);
// 4. پیدا کردن یا ایجاد کیف پول
var wallet = await _context.UserWallets
.FirstOrDefaultAsync(w => w.UserId == request.UserId, cancellationToken);
if (wallet == null)
{
_logger.LogError("Wallet not found for UserId: {UserId}", request.UserId);
throw new NotFoundException($"کیف پول کاربر {request.UserId} یافت نشد");
}
// 6. ثبت تراکنش
var transaction = new Transaction
{
Amount = request.Amount,
Description = $"پرداخت دستی عضویت - {manualPayment.Description} - مرجع: {request.ReferenceNumber}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.Now,
RefId = request.ReferenceNumber,
Type = TransactionType.DepositExternal1
};
_context.Transactions.Add(transaction);
await _context.SaveChangesAsync(cancellationToken);
// 7. اعمال تغییرات بر کیف پول
var oldBalance = wallet.Balance;
var oldDiscountBalance = wallet.DiscountBalance;
wallet.Balance += request.Amount;
wallet.DiscountBalance += request.Amount;
// 8. ثبت لاگ Balance
var balanceLog = new UserWalletChangeLog
{
WalletId = wallet.Id,
CurrentBalance = wallet.Balance,
ChangeValue = request.Amount,
CurrentNetworkBalance = wallet.NetworkBalance,
ChangeNerworkValue = 0,
CurrentDiscountBalance = wallet.DiscountBalance,
ChangeDiscountValue = request.Amount,
IsIncrease = true,
RefrenceId = transaction.Id
};
await _context.UserWalletChangeLogs.AddAsync(balanceLog, cancellationToken);
// 10. به‌روزرسانی ManualPayment با TransactionId
manualPayment.TransactionId = transaction.Id;
await _context.SaveChangesAsync(cancellationToken);
// 11. پیدا کردن یا ایجاد آدرس پیشفرض کاربر
var userAddress = await _context.UserAddresses
.Where(a => a.UserId == request.UserId)
.OrderByDescending(a => a.IsDefault)
.ThenBy(a => a.Id)
.FirstOrDefaultAsync(cancellationToken);
if (userAddress == null)
{
userAddress = new UserAddress
{
UserId = request.UserId,
Title = "آدرس پیشفرض",
Address = "پرداخت دستی عضویت - آدرس موقت",
PostalCode = "0000000000",
IsDefault = true,
CityId = 1
};
await _context.UserAddresses.AddAsync(userAddress, cancellationToken);
await _context.SaveChangesAsync(cancellationToken);
}
// 12. ثبت سفارش
var order = new UserOrder
{
UserId = request.UserId,
Amount = request.Amount,
TransactionId = transaction.Id,
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.Now,
PaymentMethod = PaymentMethod.Deposit,
DeliveryStatus = DeliveryStatus.None,
UserAddressId = userAddress.Id,
DeliveryDescription = $"پرداخت دستی عضویت - مرجع: {request.ReferenceNumber}"
};
_context.UserOrders.Add(order);
await _context.SaveChangesAsync(cancellationToken);
_logger.LogInformation(
"Manual membership payment processed successfully. UserId: {UserId}, Amount: {Amount}, ManualPaymentId: {ManualPaymentId}, TransactionId: {TransactionId}, OrderId: {OrderId}, AdminUserId: {AdminUserId}",
request.UserId, request.Amount, manualPayment.Id, transaction.Id, order.Id, adminUserId);
return new ProcessManualMembershipPaymentResponseDto
{
TransactionId = transaction.Id,
OrderId = order.Id,
NewWalletBalance = wallet.Balance,
Message = "پرداخت دستی با موفقیت ثبت شد"
};
}
catch (Exception ex) when (ex is not NotFoundException)
{
_logger.LogError(ex,
"Error processing manual membership payment for UserId: {UserId}, Amount: {Amount}",
request.UserId, request.Amount);
throw;
}
}
}

View File

@@ -0,0 +1,21 @@
namespace CMSMicroservice.Application.ManualPaymentCQ.Commands.ProcessManualMembershipPayment;
public class ProcessManualMembershipPaymentCommandValidator : AbstractValidator<ProcessManualMembershipPaymentCommand>
{
public ProcessManualMembershipPaymentCommandValidator()
{
RuleFor(x => x.UserId)
.GreaterThan(0)
.WithMessage("شناسه کاربر معتبر نیست");
RuleFor(x => x.Amount)
.GreaterThan(0)
.WithMessage("مبلغ باید بزرگتر از صفر باشد");
RuleFor(x => x.ReferenceNumber)
.NotEmpty()
.WithMessage("شماره مرجع الزامی است")
.MaximumLength(50)
.WithMessage("شماره مرجع نباید بیشتر از 50 کاراکتر باشد");
}
}

View File

@@ -0,0 +1,24 @@
namespace CMSMicroservice.Application.ManualPaymentCQ.Commands.ProcessManualMembershipPayment;
public class ProcessManualMembershipPaymentResponseDto
{
/// <summary>
/// شناسه تراکنش ثبت شده
/// </summary>
public long TransactionId { get; set; }
/// <summary>
/// شناسه سفارش ثبت شده
/// </summary>
public long OrderId { get; set; }
/// <summary>
/// موجودی جدید کیف پول
/// </summary>
public long NewWalletBalance { get; set; }
/// <summary>
/// پیام موفقیت
/// </summary>
public string Message { get; set; } = "پرداخت دستی با موفقیت ثبت شد";
}

View File

@@ -71,7 +71,7 @@ public class RejectManualPaymentCommandHandler : IRequestHandler<RejectManualPay
// 4. رد درخواست
manualPayment.Status = ManualPaymentStatus.Rejected;
manualPayment.ApprovedBy = rejectedById;
manualPayment.ApprovedAt = DateTime.UtcNow;
manualPayment.ApprovedAt = DateTime.Now;
manualPayment.RejectionReason = request.RejectionReason;
await _context.SaveChangesAsync(cancellationToken);

View File

@@ -11,7 +11,7 @@ public record GetNetworkMembershipHistoryQuery : IRequest<GetNetworkMembershipHi
public long? UserId { get; init; }
/// <summary>
/// مرتب‌سازی (پیش‌فرض: -Created)
/// مرتب‌سازی (پیش‌فرض: Created)
/// </summary>
public string? SortBy { get; init; }

View File

@@ -20,7 +20,7 @@ public class GetNetworkMembershipHistoryQueryHandler : IRequestHandler<GetNetwor
query = query.Where(x => x.UserId == request.UserId.Value);
}
query = query.ApplyOrder(sortBy: request.SortBy ?? "-Created");
query = query.ApplyOrder(sortBy: request.SortBy ?? "Created");
var meta = await query.GetMetaData(request.PaginationState, cancellationToken);

View File

@@ -55,7 +55,7 @@ public class GetNetworkStatisticsQueryHandler : IRequestHandler<GetNetworkStatis
}
// Monthly growth (last 6 months) - using Created date
var sixMonthsAgo = DateTime.UtcNow.AddMonths(-6);
var sixMonthsAgo = DateTime.Now.AddMonths(-6);
var monthlyGrowth = await _context.Users
.Where(x => x.NetworkParentId != null && x.Created >= sixMonthsAgo)
.GroupBy(x => new { x.Created.Year, x.Created.Month })

View File

@@ -9,8 +9,8 @@ public class GetNetworkTreeQueryValidator : AbstractValidator<GetNetworkTreeQuer
.WithMessage("شناسه کاربر معتبر نیست");
RuleFor(x => x.MaxDepth)
.InclusiveBetween(1, 10)
.WithMessage("عمق درخت باید بین 1 تا 10 باشد");
.InclusiveBetween(1, 20)
.WithMessage("عمق درخت باید بین 1 تا 20 باشد");
}
public Func<object, string, Task<IEnumerable<string>>> ValidateValue => async (model, propertyName) =>

View File

@@ -11,6 +11,7 @@ public class GetUserNetworkPositionQueryHandler : IRequestHandler<GetUserNetwork
public async Task<UserNetworkPositionDto?> Handle(GetUserNetworkPositionQuery request, CancellationToken cancellationToken)
{
// واکشی اطلاعات اصلی کاربر
var user = await _context.Users
.AsNoTracking()
.Where(x => x.Id == request.UserId)
@@ -20,8 +21,17 @@ public class GetUserNetworkPositionQueryHandler : IRequestHandler<GetUserNetwork
x.Mobile,
x.FirstName,
x.LastName,
x.Email,
x.NationalCode,
x.ReferralCode,
x.IsMobileVerified,
x.BirthDate,
x.NetworkParentId,
x.LegPosition
x.LegPosition,
x.HasReceivedDayaCredit,
x.DayaCreditReceivedAt,
x.PackagePurchaseMethod,
x.Created
})
.FirstOrDefaultAsync(cancellationToken);
@@ -30,41 +40,196 @@ public class GetUserNetworkPositionQueryHandler : IRequestHandler<GetUserNetwork
return null;
}
// شمارش فرزندان
var childrenCount = await _context.Users
.CountAsync(x => x.NetworkParentId == request.UserId, cancellationToken);
var leftChildCount = await _context.Users
.CountAsync(x => x.NetworkParentId == request.UserId && x.LegPosition == NetworkLeg.Left,
cancellationToken);
var rightChildCount = await _context.Users
.CountAsync(x => x.NetworkParentId == request.UserId && x.LegPosition == NetworkLeg.Right,
cancellationToken);
// اطلاعات والد
string? parentMobile = null;
string? parentFullName = null;
if (user.NetworkParentId.HasValue)
{
parentMobile = await _context.Users
var parent = await _context.Users
.Where(x => x.Id == user.NetworkParentId)
.Select(x => x.Mobile)
.Select(x => new { x.Mobile, x.FirstName, x.LastName })
.FirstOrDefaultAsync(cancellationToken);
if (parent != null)
{
parentMobile = parent.Mobile;
parentFullName = $"{parent.FirstName} {parent.LastName}".Trim();
}
}
// شمارش فرزندان مستقیم
var directChildren = await _context.Users
.Where(x => x.NetworkParentId == request.UserId)
.Select(x => new { x.Id, x.LegPosition, x.FirstName, x.LastName, x.Mobile, x.Created })
.ToListAsync(cancellationToken);
var leftChild = directChildren.FirstOrDefault(x => x.LegPosition == NetworkLeg.Left);
var rightChild = directChildren.FirstOrDefault(x => x.LegPosition == NetworkLeg.Right);
// محاسبه تعداد کل اعضای هر شاخه (با استفاده از CTE برای عملکرد بهتر)
var leftLegCount = await GetLegMemberCountAsync(request.UserId, NetworkLeg.Left, cancellationToken);
var rightLegCount = await GetLegMemberCountAsync(request.UserId, NetworkLeg.Right, cancellationToken);
var totalNetworkSize = leftLegCount + rightLegCount;
// محاسبه حداکثر عمق شبکه
var maxDepth = await GetMaxNetworkDepthAsync(request.UserId, cancellationToken);
// آمار اعضای فعال/غیرفعال (کسانی که پکیج خریده‌اند)
var allDescendantIds = await GetAllDescendantIdsAsync(request.UserId, cancellationToken);
var activeCount = await _context.Users
.CountAsync(x => allDescendantIds.Contains(x.Id) &&
x.PackagePurchaseMethod != PackagePurchaseMethod.None,
cancellationToken);
var inactiveCount = allDescendantIds.Count - activeCount;
// آمار کمیسیون‌های کاربر
var commissionStats = await _context.UserCommissionPayouts
.Where(x => x.UserId == request.UserId)
.GroupBy(x => x.UserId)
.Select(g => new
{
TotalBalances = g.Sum(x => x.BalancesEarned),
TotalAmount = g.Sum(x => x.TotalAmount),
PaidAmount = g.Where(x => x.Status == CommissionPayoutStatus.Paid).Sum(x => x.TotalAmount),
PendingAmount = g.Where(x => x.Status == CommissionPayoutStatus.Pending).Sum(x => x.TotalAmount)
})
.FirstOrDefaultAsync(cancellationToken);
return new UserNetworkPositionDto
{
// اطلاعات اصلی
UserId = user.Id,
Mobile = user.Mobile,
FirstName = user.FirstName,
LastName = user.LastName,
Email = user.Email,
NationalCode = user.NationalCode,
ReferralCode = user.ReferralCode,
IsMobileVerified = user.IsMobileVerified,
BirthDate = user.BirthDate,
JoinedAt = user.Created,
// اطلاعات شبکه
NetworkParentId = user.NetworkParentId,
ParentMobile = parentMobile,
ParentFullName = parentFullName,
LegPosition = user.LegPosition,
TotalChildren = childrenCount,
LeftChildCount = leftChildCount,
RightChildCount = rightChildCount,
IsInNetwork = user.NetworkParentId.HasValue
IsInNetwork = user.NetworkParentId.HasValue,
// آمار فرزندان مستقیم
TotalChildren = directChildren.Count,
LeftChildCount = leftChild != null ? 1 : 0,
RightChildCount = rightChild != null ? 1 : 0,
// اطلاعات فرزند چپ
LeftChildId = leftChild?.Id,
LeftChildFullName = leftChild != null ? $"{leftChild.FirstName} {leftChild.LastName}".Trim() : null,
LeftChildMobile = leftChild?.Mobile,
LeftChildJoinedAt = leftChild?.Created,
// اطلاعات فرزند راست
RightChildId = rightChild?.Id,
RightChildFullName = rightChild != null ? $"{rightChild.FirstName} {rightChild.LastName}".Trim() : null,
RightChildMobile = rightChild?.Mobile,
RightChildJoinedAt = rightChild?.Created,
// آمار کل شبکه
TotalLeftLegMembers = leftLegCount,
TotalRightLegMembers = rightLegCount,
TotalNetworkSize = totalNetworkSize,
MaxNetworkDepth = maxDepth,
// اطلاعات پکیج و دایا
HasReceivedDayaCredit = user.HasReceivedDayaCredit,
DayaCreditReceivedAt = user.DayaCreditReceivedAt,
PackagePurchaseMethod = user.PackagePurchaseMethod,
HasPurchasedGoldenPackage = user.PackagePurchaseMethod != PackagePurchaseMethod.None,
// آمار مالی
TotalEarnedCommission = commissionStats?.TotalAmount ?? 0,
TotalPaidCommission = commissionStats?.PaidAmount ?? 0,
PendingCommission = commissionStats?.PendingAmount ?? 0,
TotalBalancesEarned = commissionStats?.TotalBalances ?? 0,
// آمار فعالیت
ActiveMembersInNetwork = activeCount,
InactiveMembersInNetwork = inactiveCount
};
}
/// <summary>
/// محاسبه تعداد اعضای یک شاخه (چپ یا راست) به صورت بازگشتی
/// </summary>
private async Task<int> GetLegMemberCountAsync(long userId, NetworkLeg leg, CancellationToken cancellationToken)
{
var directChild = await _context.Users
.Where(x => x.NetworkParentId == userId && x.LegPosition == leg)
.Select(x => x.Id)
.FirstOrDefaultAsync(cancellationToken);
if (directChild == 0)
return 0;
// تعداد کل زیرمجموعه این فرزند + خود فرزند
var descendants = await GetAllDescendantIdsAsync(directChild, cancellationToken);
return descendants.Count + 1; // +1 برای خود فرزند
}
/// <summary>
/// محاسبه حداکثر عمق شبکه
/// </summary>
private async Task<int> GetMaxNetworkDepthAsync(long userId, CancellationToken cancellationToken)
{
var maxDepth = 0;
var currentLevelIds = new List<long> { userId };
while (currentLevelIds.Any())
{
var nextLevelIds = await _context.Users
.Where(x => currentLevelIds.Contains(x.NetworkParentId ?? 0))
.Select(x => x.Id)
.ToListAsync(cancellationToken);
if (nextLevelIds.Any())
{
maxDepth++;
currentLevelIds = nextLevelIds;
}
else
{
break;
}
}
return maxDepth;
}
/// <summary>
/// دریافت تمام ID های زیرمجموعه یک کاربر
/// </summary>
private async Task<List<long>> GetAllDescendantIdsAsync(long userId, CancellationToken cancellationToken)
{
var allDescendants = new List<long>();
var currentLevelIds = new List<long> { userId };
while (currentLevelIds.Any())
{
var children = await _context.Users
.Where(x => currentLevelIds.Contains(x.NetworkParentId ?? 0))
.Select(x => x.Id)
.ToListAsync(cancellationToken);
if (children.Any())
{
allDescendants.AddRange(children);
currentLevelIds = children;
}
else
{
break;
}
}
return allDescendants;
}
}

View File

@@ -5,15 +5,62 @@ namespace CMSMicroservice.Application.NetworkMembershipCQ.Queries.GetUserNetwork
/// </summary>
public class UserNetworkPositionDto
{
// اطلاعات اصلی کاربر
public long UserId { get; set; }
public string? Mobile { get; set; }
public string? FirstName { get; set; }
public string? LastName { get; set; }
public string? Email { get; set; }
public string? NationalCode { get; set; }
public string ReferralCode { get; set; } = string.Empty;
public bool IsMobileVerified { get; set; }
public DateTime? BirthDate { get; set; }
public DateTime JoinedAt { get; set; }
// اطلاعات شبکه
public long? NetworkParentId { get; set; }
public string? ParentMobile { get; set; }
public string? ParentFullName { get; set; }
public NetworkLeg? LegPosition { get; set; }
public bool IsInNetwork { get; set; }
// آمار فرزندان مستقیم
public int TotalChildren { get; set; }
public int LeftChildCount { get; set; }
public int RightChildCount { get; set; }
public bool IsInNetwork { get; set; }
// آمار کل زیرمجموعه (تمام سطوح)
public int TotalLeftLegMembers { get; set; }
public int TotalRightLegMembers { get; set; }
public int TotalNetworkSize { get; set; }
// اطلاعات فرزندان مستقیم
public long? LeftChildId { get; set; }
public string? LeftChildFullName { get; set; }
public string? LeftChildMobile { get; set; }
public DateTime? LeftChildJoinedAt { get; set; }
public long? RightChildId { get; set; }
public string? RightChildFullName { get; set; }
public string? RightChildMobile { get; set; }
public DateTime? RightChildJoinedAt { get; set; }
// اطلاعات پکیج و دایا
public bool HasReceivedDayaCredit { get; set; }
public DateTime? DayaCreditReceivedAt { get; set; }
public PackagePurchaseMethod PackagePurchaseMethod { get; set; }
public bool HasPurchasedGoldenPackage { get; set; }
// آمار مالی
public decimal TotalEarnedCommission { get; set; }
public decimal TotalPaidCommission { get; set; }
public decimal PendingCommission { get; set; }
public int TotalBalancesEarned { get; set; }
// عمق شبکه
public int MaxNetworkDepth { get; set; }
// آمار فعالیت
public int ActiveMembersInNetwork { get; set; }
public int InactiveMembersInNetwork { get; set; }
}

View File

@@ -125,7 +125,7 @@ public class VerifyGoldenPackagePurchaseCommandHandler : IRequestHandler<VerifyG
Amount = order.Amount,
Description = $"خرید پکیج طلایی از درگاه - سفارش #{order.Id}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
RefId = verifyResult.RefId,
Type = TransactionType.DepositIpg
};
@@ -152,7 +152,7 @@ public class VerifyGoldenPackagePurchaseCommandHandler : IRequestHandler<VerifyG
// 7. به‌روزرسانی سفارش و کاربر
order.TransactionId = transaction.Id;
order.PaymentStatus = PaymentStatus.Success;
order.PaymentDate = DateTime.UtcNow;
order.PaymentDate = DateTime.Now;
order.PaymentMethod = PaymentMethod.IPG;
order.User.PackagePurchaseMethod = PackagePurchaseMethod.DirectPurchase;

View File

@@ -116,7 +116,7 @@ public class VerifyPackagePurchaseCommandHandler
Amount = order.Amount,
Description = $"خرید پکیج از درگاه - سفارش #{order.Id}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
RefId = verifyResult.RefId,
Type = TransactionType.DepositIpg
};
@@ -157,7 +157,7 @@ public class VerifyPackagePurchaseCommandHandler
// 8. به‌روزرسانی Order
order.TransactionId = transaction.Id;
order.PaymentStatus = PaymentStatus.Success;
order.PaymentDate = DateTime.UtcNow;
order.PaymentDate = DateTime.Now;
order.PaymentMethod = PaymentMethod.IPG;
// 9. تغییر User.PackagePurchaseMethod

View File

@@ -31,7 +31,7 @@ public class ArchiveMessageCommandHandler : IRequestHandler<ArchiveMessageComman
// 3. آرشیو کردن:
// - message.IsArchived = true
// - message.IsActive = false // غیرفعال هم می‌شود
// - message.ArchivedAt = DateTime.UtcNow
// - message.ArchivedAt = DateTime.Now
//
// 4. ذخیره و Log:
// - await _context.SaveChangesAsync(cancellationToken)

View File

@@ -26,7 +26,7 @@ public class CreatePublicMessageCommandValidator : AbstractValidator<CreatePubli
RuleFor(x => x.ExpiresAt)
.NotEmpty().WithMessage("تاریخ پایان الزامی است")
.GreaterThan(DateTime.UtcNow).WithMessage("تاریخ پایان باید در آینده باشد");
.GreaterThan(DateTime.Now).WithMessage("تاریخ پایان باید در آینده باشد");
RuleFor(x => x.LinkUrl)
.MaximumLength(500).WithMessage("لینک نمی‌تواند بیشتر از 500 کاراکتر باشد")

View File

@@ -33,10 +33,10 @@ public class PublishMessageCommandHandler : IRequestHandler<PublishMessageComman
//
// 3. فعال‌سازی پیام:
// - message.IsActive = true
// - message.PublishedAt = DateTime.UtcNow
// - message.PublishedAt = DateTime.Now
// - اگر StartDate خالی است، از الان شروع کن:
// if (!message.StartDate.HasValue)
// message.StartDate = DateTime.UtcNow
// message.StartDate = DateTime.Now
//
// 4. ذخیره و Log:
// - await _context.SaveChangesAsync(cancellationToken)

View File

@@ -21,7 +21,7 @@ public class GetActiveMessagesQueryHandler : IRequestHandler<GetActiveMessagesQu
public async Task<List<PublicMessageDto>> Handle(GetActiveMessagesQuery request, CancellationToken cancellationToken)
{
var now = DateTime.UtcNow;
var now = DateTime.Now;
var query = _context.PublicMessages
.Where(x => !x.IsDeleted

View File

@@ -22,7 +22,7 @@ public class GetAllMessagesQueryHandler : IRequestHandler<GetAllMessagesQuery, G
public async Task<GetAllMessagesResponseDto> Handle(GetAllMessagesQuery request, CancellationToken cancellationToken)
{
var now = DateTime.UtcNow;
var now = DateTime.Now;
// Query پایه
var query = _context.PublicMessages

View File

@@ -43,7 +43,7 @@ public class RefundTransactionCommandHandler : IRequestHandler<RefundTransaction
Amount = -refundAmount, // مبلغ منفی برای استرداد
Description = $"استرداد تراکنش {request.TransactionId}: {request.RefundReason}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
RefId = $"REFUND-{originalTransaction.RefId}",
Type = TransactionType.Buy // یا می‌تونیم یک نوع جدید برای Refund تعریف کنیم
};

View File

@@ -10,6 +10,8 @@ public record GetAllUserByFilterQuery : IRequest<GetAllUserByFilterResponseDto>
}public class GetAllUserByFilterFilter
{
//جستجوی متنی (نام، نام خانوادگی، موبایل، کدملی)
public string? SearchText { get; set; }
//شناسه
public long? Id { get; set; }
//نام

View File

@@ -17,6 +17,11 @@ public class GetAllUserByFilterQueryHandler : IRequestHandler<GetAllUserByFilter
if (request.Filter is not null)
{
query = query
.Where(x => request.Filter.SearchText == null ||
x.FirstName.Contains(request.Filter.SearchText) ||
x.LastName.Contains(request.Filter.SearchText) ||
x.Mobile.Contains(request.Filter.SearchText) ||
x.NationalCode.Contains(request.Filter.SearchText))
.Where(x => request.Filter.Id == null || x.Id == request.Filter.Id)
.Where(x => request.Filter.FirstName == null || x.FirstName.Contains(request.Filter.FirstName))
.Where(x => request.Filter.LastName == null || x.LastName.Contains(request.Filter.LastName))

View File

@@ -49,7 +49,7 @@ public class ApplyDiscountToOrderCommandHandler : IRequestHandler<ApplyDiscountT
// DiscountAmount = request.DiscountAmount,
// Reason = request.Reason,
// DiscountCode = request.DiscountCode,
// AppliedAt = DateTime.UtcNow
// AppliedAt = DateTime.Now
// }
// - await _context.OrderDiscountLogs.AddAsync(discountLog, cancellationToken)
//

View File

@@ -50,7 +50,7 @@ public class CancelOrderCommandHandler : IRequestHandler<CancelOrderCommand, Can
Amount = -order.Amount,
Description = $"بازگشت وجه سفارش {request.OrderId}: {request.CancelReason}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
RefId = $"REFUND-ORDER-{order.Id}",
Type = TransactionType.Buy
};

View File

@@ -147,7 +147,7 @@ public class
VATAmount = vatAmount,
TotalAmount = totalAmount,
IsPaid = true,
PaidAt = DateTime.UtcNow
PaidAt = DateTime.Now
};
await _context.OrderVATs.AddAsync(orderVAT, cancellationToken);

View File

@@ -9,7 +9,7 @@ public class GetOrdersByDateRangeQueryValidator : AbstractValidator<GetOrdersByD
.WithMessage("تاریخ شروع باید کوچکتر یا مساوی تاریخ پایان باشد");
RuleFor(x => x.EndDate)
.LessThanOrEqualTo(DateTime.UtcNow.AddDays(1))
.LessThanOrEqualTo(DateTime.Now.AddDays(1))
.WithMessage("تاریخ پایان نمی‌تواند در آینده باشد");
RuleFor(x => x.PageIndex)

View File

@@ -92,7 +92,7 @@ public class VerifyDiscountWalletChargeCommandHandler
Amount = request.Amount,
Description = $"شارژ کیف پول تخفیفی - کاربر {user.Id}",
PaymentStatus = PaymentStatus.Success,
PaymentDate = DateTime.UtcNow,
PaymentDate = DateTime.Now,
RefId = verifyResult.RefId,
Type = TransactionType.DiscountWalletCharge
};

View File

@@ -40,6 +40,11 @@ public class UserClubFeature : BaseAuditableEntity
/// </summary>
public DateTime GrantedAt { get; set; }
/// <summary>
/// وضعیت فعال/غیرفعال بودن ویژگی برای کاربر (قابل مدیریت توسط ادمین)
/// </summary>
public bool IsActive { get; set; } = true;
/// <summary>
/// یادداشت اختیاری
/// </summary>

View File

@@ -65,6 +65,23 @@ public class NetworkWeeklyBalance : BaseAuditableEntity
/// </summary>
public int RightLegRemainder { get; set; }
/// <summary>
/// مقدار فلش هر طرف (بعد از اعمال Cap): TotalBalances - CappedBalances
/// </summary>
public int FlushedPerSide { get; set; }
/// <summary>
/// مجموع فلش از دو طرف: FlushedPerSide × 2
/// این مقدار از دست می‌رود (نه به هفته بعد، نه به کمیسیون)
/// </summary>
public int TotalFlushed { get; set; }
/// <summary>
/// مجموع تعادل زیرمجموعه این کاربر (تا 15 لول پایین‌تر)
/// این مقدار در CalculateWeeklyBalances محاسبه و ذخیره می‌شود
/// </summary>
public int SubordinateBalances { get; set; }
/// <summary>
/// [DEPRECATED] تعداد تعادل شاخه چپ - استفاده نشود
/// </summary>

View File

@@ -6,17 +6,28 @@ namespace CMSMicroservice.Domain.Enums;
public enum DayaLoanStatus
{
/// <summary>
/// در انتظار دریافت وام (خرید انجام شده، قرارداد امضا شده، درخواست وام ثبت شده)
/// هنوز درخواست نشده
/// </summary>
PendingReceive = 0,
NotRequested = 0,
/// <summary>
/// وام دریافت شده (در آینده اضافه می‌شود)
/// در انتظار دریافت وام (قرارداد امضا شده، در انتظار تسویه)
/// وضعیت: "فعال شده (در انتظار تسویه)"
/// </summary>
Received = 1,
PendingReceive = 1,
/// <summary>
/// رد شده (در آینده اضافه می‌شود)
/// وام دریافت شده و شارژ شده
/// </summary>
Rejected = 2,
Received = 2,
/// <summary>
/// رد شده
/// </summary>
Rejected = 3,
/// <summary>
/// در حال بررسی
/// </summary>
UnderReview = 4,
}

View File

@@ -4,4 +4,5 @@ public enum PaymentMethod
{
IPG = 0,
Wallet = 1,
Deposit = 2,
}

View File

@@ -1,6 +1,7 @@
using CMSMicroservice.Application.CommissionCQ.Commands.CalculateWeeklyBalances;
using CMSMicroservice.Application.CommissionCQ.Commands.CalculateWeeklyCommissionPool;
using CMSMicroservice.Application.CommissionCQ.Commands.ProcessUserPayouts;
using CMSMicroservice.Application.CommissionCQ.Commands.TriggerWeeklyCalculation;
using CMSMicroservice.Application.Common.Interfaces;
using CMSMicroservice.Domain.Entities;
using CMSMicroservice.Domain.Enums;
@@ -54,16 +55,30 @@ public class WeeklyCommissionJob
/// <summary>
/// Execute weekly commission calculation with retry logic
/// Called by Hangfire scheduler
/// Called by Hangfire scheduler or manually triggered
/// </summary>
public async Task ExecuteAsync(CancellationToken cancellationToken = default)
/// <param name="weekNumber">Week number in YYYY-Www format (e.g., 2025-W48). If null, uses previous week.</param>
/// <param name="cancellationToken">Cancellation token</param>
public async Task ExecuteAsync(string? weekNumber = null, CancellationToken cancellationToken = default)
{
var executionId = Guid.NewGuid();
var startTime = DateTime.UtcNow;
var startTime = DateTime.Now;
// Calculate for PREVIOUS week (completed week)
var previousWeek = DateTime.UtcNow.AddDays(-7);
var previousWeekNumber = GetWeekNumber(previousWeek);
// Use provided week number or calculate for PREVIOUS week (completed week)
string targetWeekNumber;
if (!string.IsNullOrWhiteSpace(weekNumber))
{
targetWeekNumber = weekNumber;
_logger.LogInformation("📅 Using manually specified week: {WeekNumber}", targetWeekNumber);
}
else
{
var previousWeek = DateTime.Now.AddDays(-7);
targetWeekNumber = GetWeekNumber(previousWeek);
_logger.LogInformation("📅 Using previous week (auto-calculated): {WeekNumber}", targetWeekNumber);
}
var previousWeekNumber = targetWeekNumber;
_logger.LogInformation(
"🚀 [{ExecutionId}] Starting weekly commission calculation for {WeekNumber}",
@@ -89,7 +104,7 @@ public class WeeklyCommissionJob
}, cancellationToken);
// Update log on success
var completedAt = DateTime.UtcNow;
var completedAt = DateTime.Now;
var duration = completedAt - startTime;
log.Status = WorkerExecutionStatus.Success;
@@ -113,7 +128,7 @@ public class WeeklyCommissionJob
catch (Exception ex)
{
// Update log on failure
var completedAt = DateTime.UtcNow;
var completedAt = DateTime.Now;
var duration = completedAt - startTime;
log.Status = WorkerExecutionStatus.Failed;
@@ -165,38 +180,43 @@ public class WeeklyCommissionJob
"📊 [{ExecutionId}] Step 1/3: Calculating weekly balances...",
executionId);
await _mediator.Send(new CalculateWeeklyBalancesCommand
await _mediator.Send(new TriggerWeeklyCalculationCommand
{
WeekNumber = weekNumber,
ForceRecalculate = false
}, cancellationToken);
// Step 2: Calculate global commission pool
_logger.LogInformation(
"💰 [{ExecutionId}] Step 2/3: Calculating commission pool...",
executionId);
await _mediator.Send(new CalculateWeeklyCommissionPoolCommand
{
WeekNumber = weekNumber,
ForceRecalculate = false
}, cancellationToken);
// Step 3: Distribute commissions to users
_logger.LogInformation(
"💸 [{ExecutionId}] Step 3/3: Processing user payouts...",
executionId);
await _mediator.Send(new ProcessUserPayoutsCommand
{
WeekNumber = weekNumber,
ForceReprocess = false
}, cancellationToken);
// await _mediator.Send(new CalculateWeeklyBalancesCommand
// {
// WeekNumber = weekNumber,
// ForceRecalculate = false
// }, cancellationToken);
//
// // Step 2: Calculate global commission pool
// _logger.LogInformation(
// "💰 [{ExecutionId}] Step 2/3: Calculating commission pool...",
// executionId);
//
// await _mediator.Send(new CalculateWeeklyCommissionPoolCommand
// {
// WeekNumber = weekNumber,
// ForceRecalculate = false
// }, cancellationToken);
//
// // Step 3: Distribute commissions to users
// _logger.LogInformation(
// "💸 [{ExecutionId}] Step 3/3: Processing user payouts...",
// executionId);
//
// await _mediator.Send(new ProcessUserPayoutsCommand
// {
// WeekNumber = weekNumber,
// ForceReprocess = false
// }, cancellationToken);
transaction.Complete();
_logger.LogInformation(
"✅ [{ExecutionId}] All 3 steps completed successfully",
"✅ [{ExecutionId}] All 2 steps completed successfully",
executionId);
}
catch (Exception ex)
@@ -216,8 +236,8 @@ public class WeeklyCommissionJob
var calendar = System.Globalization.CultureInfo.InvariantCulture.Calendar;
var weekNumber = calendar.GetWeekOfYear(
date,
System.Globalization.CalendarWeekRule.FirstFourDayWeek,
DayOfWeek.Monday);
System.Globalization.CalendarWeekRule.FirstDay,
DayOfWeek.Saturday);
var year = date.Year;
if (weekNumber >= 52 && date.Month == 1)

View File

@@ -109,7 +109,7 @@ public class WeeklyNetworkCommissionWorker : BackgroundService
private async Task ExecuteWeeklyCalculationAsync(CancellationToken cancellationToken)
{
var executionId = Guid.NewGuid();
var startTime = DateTime.UtcNow;
var startTime = DateTime.Now;
_logger.LogInformation("=== Starting Weekly Commission Calculation [{ExecutionId}] at {Time} (UTC) ===",
executionId, startTime);
@@ -154,7 +154,7 @@ public class WeeklyNetworkCommissionWorker : BackgroundService
// Update log
log.Status = WorkerExecutionStatus.SuccessWithWarnings;
log.CompletedAt = DateTime.UtcNow;
log.CompletedAt = DateTime.Now;
log.DurationMs = (long)(log.CompletedAt.Value - log.StartedAt).TotalMilliseconds;
log.Details = "Week already calculated - skipped";
await context.SaveChangesAsync(cancellationToken);
@@ -222,7 +222,7 @@ public class WeeklyNetworkCommissionWorker : BackgroundService
// Commit Transaction
transaction.Complete();
var completedAt = DateTime.UtcNow;
var completedAt = DateTime.Now;
var duration = completedAt - startTime;
// Update log - Success
@@ -297,7 +297,7 @@ public class WeeklyNetworkCommissionWorker : BackgroundService
var context = errorScope.ServiceProvider.GetRequiredService<IApplicationDbContext>();
log.Status = WorkerExecutionStatus.Failed;
log.CompletedAt = DateTime.UtcNow;
log.CompletedAt = DateTime.Now;
log.DurationMs = (long)(log.CompletedAt.Value - log.StartedAt).TotalMilliseconds;
log.ErrorCount = 1;
log.ErrorMessage = ex.Message;

View File

@@ -32,7 +32,39 @@ public static class ConfigureServices
services.AddScoped<INetworkPlacementService, NetworkPlacementService>();
services.AddScoped<IAlertService, AlertService>();
services.AddScoped<IUserNotificationService, UserNotificationService>();
services.AddScoped<IDayaLoanApiService, MockDayaLoanApiService>(); // Mock - جایگزین با Real برای Production
// Daya Loan API Service - قابل تغییر بین Mock و Real
var useMockDayaApi = configuration.GetValue<bool>("DayaApi:UseMock", false);
if (useMockDayaApi)
{
// Mock برای Development/Testing
services.AddScoped<IDayaLoanApiService, MockDayaLoanApiService>();
}
else
{
// Real Implementation با HttpClient
services.AddHttpClient<IDayaLoanApiService, DayaLoanApiService>()
.SetHandlerLifetime(TimeSpan.FromMinutes(5))
.ConfigureHttpClient((sp, client) =>
{
var config = sp.GetRequiredService<IConfiguration>();
// Base Address
var baseAddress = config["DayaApi:BaseAddress"] ?? "https://testdaya.tadbirandishan.com";
client.BaseAddress = new Uri(baseAddress);
// Merchant Permission Key
var permissionKey = config["DayaApi:MerchantPermissionKey"];
if (!string.IsNullOrEmpty(permissionKey))
{
client.DefaultRequestHeaders.Add("merchant-permission-key", permissionKey);
}
// Timeout
client.Timeout = TimeSpan.FromSeconds(30);
});
}
// Payment Gateway Service - فقط Daya (درگاه اینترنتی از Gateway میاد نه CMS)
var useRealPaymentGateway = configuration.GetValue<bool>("UseRealPaymentGateway", false);

View File

@@ -6,13 +6,13 @@
BEGIN TRANSACTION;
-- Step 1: Validation - Find users with more than 2 children (INVALID for binary tree)
-- Step 1: Validation - Find CMS.Users with more than 2 children (INVALID for binary tree)
-- این کاربران باید قبل از Migration بررسی شوند
SELECT
ParentId,
COUNT(*) as ChildCount,
STRING_AGG(CAST(Id AS VARCHAR), ', ') as ChildIds
FROM Users
FROM CMS.Users
WHERE ParentId IS NOT NULL
GROUP BY ParentId
HAVING COUNT(*) > 2;
@@ -20,8 +20,8 @@ HAVING COUNT(*) > 2;
-- اگر نتیجه‌ای بود، باید دستی تصمیم بگیرید کدام 2 فرزند باقی بمانند!
-- اگر نتیجه‌ای نبود، ادامه دهید:
-- Step 2: Copy ParentId → NetworkParentId for all users
UPDATE Users
-- Step 2: Copy ParentId → NetworkParentId for all CMS.Users
UPDATE CMS.Users
SET NetworkParentId = ParentId
WHERE ParentId IS NOT NULL
AND NetworkParentId IS NULL;
@@ -33,16 +33,16 @@ WITH RankedChildren AS (
Id,
ParentId,
ROW_NUMBER() OVER (PARTITION BY ParentId ORDER BY Id ASC) as ChildRank
FROM Users
FROM CMS.Users
WHERE ParentId IS NOT NULL
)
UPDATE Users
UPDATE CMS.Users
SET LegPosition = CASE
WHEN rc.ChildRank = 1 THEN 0 -- Left = 0 (enum value)
WHEN rc.ChildRank = 2 THEN 1 -- Right = 1 (enum value)
ELSE NULL -- اگر بیشتر از 2 فرزند بود (نباید اتفاق بیفته)
END
FROM Users u
FROM CMS.Users u
INNER JOIN RankedChildren rc ON u.Id = rc.Id;
-- Step 4: Validation - Check for orphaned nodes (Parent doesn't exist)
@@ -50,9 +50,9 @@ SELECT
Id,
NetworkParentId,
'Orphaned: Parent does not exist' as Issue
FROM Users
FROM CMS.Users
WHERE NetworkParentId IS NOT NULL
AND NetworkParentId NOT IN (SELECT Id FROM Users);
AND NetworkParentId NOT IN (SELECT Id FROM CMS.Users);
-- اگر Orphan یافت شد، باید آنها را NULL کنید یا Parent صحیح تخصیص دهید
@@ -62,7 +62,7 @@ SELECT
NetworkParentId,
COUNT(*) as ChildCount,
STRING_AGG(CAST(Id AS VARCHAR), ', ') as ChildIds
FROM Users
FROM CMS.Users
WHERE NetworkParentId IS NOT NULL
GROUP BY NetworkParentId
HAVING COUNT(*) > 2;
@@ -71,26 +71,26 @@ HAVING COUNT(*) > 2;
-- Step 6: Statistics
SELECT
'Total Users' as Metric,
'Total CMS.Users' as Metric,
COUNT(*) as Count
FROM Users
FROM CMS.Users
UNION ALL
SELECT
'Users with NetworkParentId',
'CMS.Users with NetworkParentId',
COUNT(*)
FROM Users
FROM CMS.Users
WHERE NetworkParentId IS NOT NULL
UNION ALL
SELECT
'Users with LegPosition Left',
'CMS.Users with LegPosition Left',
COUNT(*)
FROM Users
FROM CMS.Users
WHERE LegPosition = 0
UNION ALL
SELECT
'Users with LegPosition Right',
'CMS.Users with LegPosition Right',
COUNT(*)
FROM Users
FROM CMS.Users
WHERE LegPosition = 1;
-- Commit if validation passes

View File

@@ -1,5 +1,6 @@
using System.Reflection;
using CMSMicroservice.Application.Common.Interfaces;
using Microsoft.EntityFrameworkCore.Diagnostics;
using CMSMicroservice.Domain.Entities;
using CMSMicroservice.Domain.Entities.Payment;
@@ -39,6 +40,9 @@ public class ApplicationDbContext : DbContext, IApplicationDbContext
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.AddInterceptors(_auditableEntitySaveChangesInterceptor);
// Suppress PendingModelChangesWarning in EF Core 9
optionsBuilder.ConfigureWarnings(w => w.Ignore(RelationalEventId.PendingModelChangesWarning));
}
public override async Task<int> SaveChangesAsync(CancellationToken cancellationToken = default)

View File

@@ -32,7 +32,6 @@ public class ApplicationDbContextInitialiser
throw;
}
}
public async Task SeedAsync()
{
try

View File

@@ -20,6 +20,7 @@ public class UserClubFeatureConfiguration : IEntityTypeConfiguration<UserClubFea
builder.Property(entity => entity.ClubMembershipId).IsRequired();
builder.Property(entity => entity.ClubFeatureId).IsRequired();
builder.Property(entity => entity.GrantedAt).IsRequired();
builder.Property(entity => entity.IsActive).IsRequired().HasDefaultValue(true);
builder.Property(entity => entity.Notes).IsRequired(false).HasMaxLength(500);
// رابطه با User

View File

@@ -10,33 +10,13 @@ namespace CMSMicroservice.Infrastructure.Persistence.Migrations
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
// اضافه کردن تنظیمات Club.MembershipGiftValue
migrationBuilder.Sql(@"
INSERT INTO ""SystemConfigurations""
(""Key"", ""Value"", ""Description"", ""Scope"", ""IsActive"", ""Created"", ""CreatedBy"")
SELECT
'Club.MembershipGiftValue',
'25200000',
'مبلغ هدیه حق عضویت باشگاه (ریال) - این مبلغ از کیف پول کم نمی‌شود',
1, -- ConfigurationScope.Club = 1
true,
NOW(),
'System'
WHERE NOT EXISTS (
SELECT 1 FROM ""SystemConfigurations""
WHERE ""Key"" = 'Club.MembershipGiftValue'
);
");
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
// حذف تنظیمات Club.MembershipGiftValue
migrationBuilder.Sql(@"
DELETE FROM ""SystemConfigurations""
WHERE ""Key"" = 'Club.MembershipGiftValue';
");
}
}
}

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,80 @@
using System;
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace CMSMicroservice.Infrastructure.Persistence.Migrations
{
/// <inheritdoc />
public partial class u13 : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.AddColumn<DateTime>(
name: "ArchivedAt",
schema: "CMS",
table: "PublicMessages",
type: "datetime2",
nullable: true);
migrationBuilder.AddColumn<DateTime>(
name: "EndDate",
schema: "CMS",
table: "PublicMessages",
type: "datetime2",
nullable: true);
migrationBuilder.AddColumn<bool>(
name: "IsArchived",
schema: "CMS",
table: "PublicMessages",
type: "bit",
nullable: false,
defaultValue: false);
migrationBuilder.AddColumn<DateTime>(
name: "PublishedAt",
schema: "CMS",
table: "PublicMessages",
type: "datetime2",
nullable: true);
migrationBuilder.AddColumn<DateTime>(
name: "StartDate",
schema: "CMS",
table: "PublicMessages",
type: "datetime2",
nullable: true);
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "ArchivedAt",
schema: "CMS",
table: "PublicMessages");
migrationBuilder.DropColumn(
name: "EndDate",
schema: "CMS",
table: "PublicMessages");
migrationBuilder.DropColumn(
name: "IsArchived",
schema: "CMS",
table: "PublicMessages");
migrationBuilder.DropColumn(
name: "PublishedAt",
schema: "CMS",
table: "PublicMessages");
migrationBuilder.DropColumn(
name: "StartDate",
schema: "CMS",
table: "PublicMessages");
}
}
}

View File

@@ -0,0 +1,31 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace CMSMicroservice.Infrastructure.Persistence.Migrations
{
/// <inheritdoc />
public partial class AddIsActiveToUserClubFeatures : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.AddColumn<bool>(
name: "IsActive",
schema: "CMS",
table: "UserClubFeatures",
type: "bit",
nullable: false,
defaultValue: true);
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "IsActive",
schema: "CMS",
table: "UserClubFeatures");
}
}
}

View File

@@ -0,0 +1,44 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace CMSMicroservice.Infrastructure.Persistence.Migrations
{
/// <inheritdoc />
public partial class AddFlushedFieldsToNetworkWeeklyBalance : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.AddColumn<int>(
name: "FlushedPerSide",
schema: "CMS",
table: "NetworkWeeklyBalances",
type: "int",
nullable: false,
defaultValue: 0);
migrationBuilder.AddColumn<int>(
name: "TotalFlushed",
schema: "CMS",
table: "NetworkWeeklyBalances",
type: "int",
nullable: false,
defaultValue: 0);
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "FlushedPerSide",
schema: "CMS",
table: "NetworkWeeklyBalances");
migrationBuilder.DropColumn(
name: "TotalFlushed",
schema: "CMS",
table: "NetworkWeeklyBalances");
}
}
}

View File

@@ -0,0 +1,31 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace CMSMicroservice.Infrastructure.Persistence.Migrations
{
/// <inheritdoc />
public partial class AddSubordinateBalancesToNetworkWeeklyBalance : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.AddColumn<int>(
name: "SubordinateBalances",
schema: "CMS",
table: "NetworkWeeklyBalances",
type: "int",
nullable: false,
defaultValue: 0);
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "SubordinateBalances",
schema: "CMS",
table: "NetworkWeeklyBalances");
}
}
}

View File

@@ -204,6 +204,11 @@ namespace CMSMicroservice.Infrastructure.Persistence.Migrations
b.Property<DateTime>("GrantedAt")
.HasColumnType("datetime2");
b.Property<bool>("IsActive")
.ValueGeneratedOnAdd()
.HasColumnType("bit")
.HasDefaultValue(true);
b.Property<bool>("IsDeleted")
.HasColumnType("bit");
@@ -1272,98 +1277,6 @@ namespace CMSMicroservice.Infrastructure.Persistence.Migrations
b.ToTable("SystemConfigurationHistories", "CMS");
});
modelBuilder.Entity("CMSMicroservice.Domain.Entities.Message.PublicMessage", b =>
{
b.Property<long>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("bigint");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<long>("Id"));
b.Property<string>("Content")
.IsRequired()
.HasMaxLength(2000)
.HasColumnType("nvarchar(2000)");
b.Property<DateTime>("Created")
.HasColumnType("datetime2");
b.Property<string>("CreatedBy")
.HasColumnType("nvarchar(max)");
b.Property<long>("CreatedByUserId")
.HasColumnType("bigint");
b.Property<DateTime>("ExpiresAt")
.HasColumnType("datetime2");
b.Property<bool>("IsActive")
.ValueGeneratedOnAdd()
.HasColumnType("bit")
.HasDefaultValue(true);
b.Property<bool>("IsDeleted")
.HasColumnType("bit");
b.Property<DateTime?>("LastModified")
.HasColumnType("datetime2");
b.Property<string>("LastModifiedBy")
.HasColumnType("nvarchar(max)");
b.Property<string>("LinkText")
.HasMaxLength(100)
.HasColumnType("nvarchar(100)");
b.Property<string>("LinkUrl")
.HasMaxLength(500)
.HasColumnType("nvarchar(500)");
b.Property<int>("Priority")
.HasColumnType("int");
b.Property<DateTime>("StartsAt")
.HasColumnType("datetime2");
b.Property<string>("Title")
.IsRequired()
.HasMaxLength(200)
.HasColumnType("nvarchar(200)");
b.Property<int>("Type")
.HasColumnType("int");
b.Property<int>("ViewCount")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasDefaultValue(0);
b.HasKey("Id");
b.HasIndex("CreatedByUserId")
.HasDatabaseName("IX_PublicMessages_CreatedByUserId");
b.HasIndex("ExpiresAt")
.HasDatabaseName("IX_PublicMessages_ExpiresAt");
b.HasIndex("IsActive")
.HasDatabaseName("IX_PublicMessages_IsActive");
b.HasIndex("Priority")
.HasDatabaseName("IX_PublicMessages_Priority");
b.HasIndex("StartsAt")
.HasDatabaseName("IX_PublicMessages_StartsAt");
b.HasIndex("Type")
.HasDatabaseName("IX_PublicMessages_Type");
b.HasIndex("IsActive", "ExpiresAt")
.HasDatabaseName("IX_PublicMessages_IsActive_ExpiresAt");
b.ToTable("PublicMessages", "CMS");
});
modelBuilder.Entity("CMSMicroservice.Domain.Entities.Network.NetworkWeeklyBalance", b =>
{
b.Property<long>("Id")
@@ -1381,6 +1294,9 @@ namespace CMSMicroservice.Infrastructure.Persistence.Migrations
b.Property<string>("CreatedBy")
.HasColumnType("nvarchar(max)");
b.Property<int>("FlushedPerSide")
.HasColumnType("int");
b.Property<bool>("IsDeleted")
.HasColumnType("bit");
@@ -1423,9 +1339,15 @@ namespace CMSMicroservice.Infrastructure.Persistence.Migrations
b.Property<int>("RightLegTotal")
.HasColumnType("int");
b.Property<int>("SubordinateBalances")
.HasColumnType("int");
b.Property<int>("TotalBalances")
.HasColumnType("int");
b.Property<int>("TotalFlushed")
.HasColumnType("int");
b.Property<long>("UserId")
.HasColumnType("bigint");
@@ -1921,6 +1843,113 @@ namespace CMSMicroservice.Infrastructure.Persistence.Migrations
b.ToTable("ProductTags", "CMS");
});
modelBuilder.Entity("CMSMicroservice.Domain.Entities.PublicMessage", b =>
{
b.Property<long>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("bigint");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<long>("Id"));
b.Property<DateTime?>("ArchivedAt")
.HasColumnType("datetime2");
b.Property<string>("Content")
.IsRequired()
.HasMaxLength(2000)
.HasColumnType("nvarchar(2000)");
b.Property<DateTime>("Created")
.HasColumnType("datetime2");
b.Property<string>("CreatedBy")
.HasColumnType("nvarchar(max)");
b.Property<long>("CreatedByUserId")
.HasColumnType("bigint");
b.Property<DateTime?>("EndDate")
.HasColumnType("datetime2");
b.Property<DateTime>("ExpiresAt")
.HasColumnType("datetime2");
b.Property<bool>("IsActive")
.ValueGeneratedOnAdd()
.HasColumnType("bit")
.HasDefaultValue(true);
b.Property<bool>("IsArchived")
.HasColumnType("bit");
b.Property<bool>("IsDeleted")
.HasColumnType("bit");
b.Property<DateTime?>("LastModified")
.HasColumnType("datetime2");
b.Property<string>("LastModifiedBy")
.HasColumnType("nvarchar(max)");
b.Property<string>("LinkText")
.HasMaxLength(100)
.HasColumnType("nvarchar(100)");
b.Property<string>("LinkUrl")
.HasMaxLength(500)
.HasColumnType("nvarchar(500)");
b.Property<int>("Priority")
.HasColumnType("int");
b.Property<DateTime?>("PublishedAt")
.HasColumnType("datetime2");
b.Property<DateTime?>("StartDate")
.HasColumnType("datetime2");
b.Property<DateTime>("StartsAt")
.HasColumnType("datetime2");
b.Property<string>("Title")
.IsRequired()
.HasMaxLength(200)
.HasColumnType("nvarchar(200)");
b.Property<int>("Type")
.HasColumnType("int");
b.Property<int>("ViewCount")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasDefaultValue(0);
b.HasKey("Id");
b.HasIndex("CreatedByUserId")
.HasDatabaseName("IX_PublicMessages_CreatedByUserId");
b.HasIndex("ExpiresAt")
.HasDatabaseName("IX_PublicMessages_ExpiresAt");
b.HasIndex("IsActive")
.HasDatabaseName("IX_PublicMessages_IsActive");
b.HasIndex("Priority")
.HasDatabaseName("IX_PublicMessages_Priority");
b.HasIndex("StartsAt")
.HasDatabaseName("IX_PublicMessages_StartsAt");
b.HasIndex("Type")
.HasDatabaseName("IX_PublicMessages_Type");
b.HasIndex("IsActive", "ExpiresAt")
.HasDatabaseName("IX_PublicMessages_IsActive_ExpiresAt");
b.ToTable("PublicMessages", "CMS");
});
modelBuilder.Entity("CMSMicroservice.Domain.Entities.Role", b =>
{
b.Property<long>("Id")

View File

@@ -1,11 +1,15 @@
using CMSMicroservice.Application.DayaLoanCQ.Services;
using CMSMicroservice.Domain.Enums;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Configuration;
using System;
using System.Collections.Generic;
using System.Net.Http;
using System.Net.Http.Json;
using System.Threading;
using System.Threading.Tasks;
using System.Text.Json.Serialization;
using System.Linq;
namespace CMSMicroservice.Infrastructure.Services;
@@ -74,31 +78,232 @@ public class MockDayaLoanApiService : IDayaLoanApiService
/// <summary>
/// Real Implementation برای API واقعی دایا
/// TODO: این کلاس باید پیاده‌سازی شود زمانی که API دایا آماده شد
/// مبتنی بر مستند TA-DAYA-S10-G-MerchantServices
/// </summary>
public class DayaLoanApiService : IDayaLoanApiService
{
private readonly HttpClient _httpClient;
private readonly ILogger<DayaLoanApiService> _logger;
private readonly IConfiguration _configuration;
public DayaLoanApiService(HttpClient httpClient, ILogger<DayaLoanApiService> logger)
public DayaLoanApiService(
HttpClient httpClient,
ILogger<DayaLoanApiService> logger,
IConfiguration configuration)
{
_httpClient = httpClient;
_logger = logger;
_configuration = configuration;
// تنظیم Base Address از Configuration
var baseAddress = _configuration["DayaApi:BaseAddress"] ?? "https://testdaya.tadbirandishan.com";
_httpClient.BaseAddress = new Uri(baseAddress);
// تنظیم merchant-permission-key از Configuration
var permissionKey = _configuration["DayaApi:MerchantPermissionKey"];
if (!string.IsNullOrEmpty(permissionKey))
{
_httpClient.DefaultRequestHeaders.Add("merchant-permission-key", permissionKey);
}
else
{
_logger.LogWarning("⚠️ DayaApi:MerchantPermissionKey is not configured in appsettings!");
}
}
public async Task<List<DayaLoanStatusResult>> CheckLoanStatusAsync(
List<string> nationalCodes,
CancellationToken cancellationToken = default)
{
// TODO: پیاده‌سازی واقعی API دایا
// مثال:
// var request = new DayaApiRequest { NationalCodes = nationalCodes };
// var response = await _httpClient.PostAsJsonAsync("/api/loan/check", request, cancellationToken);
// response.EnsureSuccessStatusCode();
// var result = await response.Content.ReadFromJsonAsync<DayaApiResponse>(cancellationToken);
// return MapToResults(result);
throw new NotImplementedException("Real Daya API is not implemented yet. Use MockDayaLoanApiService for testing.");
try
{
_logger.LogInformation("Calling Daya API for {Count} national codes", nationalCodes.Count);
// ساخت Request Body
var requestBody = new DayaContractsRequest
{
NationalCodes = nationalCodes
};
// فراخوانی API
var response = await _httpClient.PostAsJsonAsync(
"/api/merchant/contracts",
requestBody,
cancellationToken);
var x = await response.Content.ReadAsStringAsync();
// خواندن پاسخ
var apiResponse = await response.Content.ReadFromJsonAsync<DayaContractsResponse>(cancellationToken);
// بررسی موفقیت
if (apiResponse == null)
{
_logger.LogError("Daya API returned null response");
return CreateEmptyResults(nationalCodes);
}
if (!apiResponse.Succeed)
{
_logger.LogWarning("Daya API call failed. Code: {Code}, Message: {Message}",
apiResponse.Code, apiResponse.Message);
return CreateEmptyResults(nationalCodes);
}
if (apiResponse.Data == null || !apiResponse.Data.Any())
{
_logger.LogInformation("Daya API returned no contract data");
return CreateEmptyResults(nationalCodes);
}
// تبدیل نتایج API به مدل داخلی
var results = MapApiResponseToResults(apiResponse.Data, nationalCodes);
_logger.LogInformation("Daya API returned {Count} contracts for {RequestCount} national codes",
results.Count(r => !string.IsNullOrEmpty(r.ContractNumber)),
nationalCodes.Count);
return results;
}
catch (HttpRequestException ex)
{
_logger.LogError(ex, "HTTP error calling Daya API. Status: {StatusCode}", ex.StatusCode);
return CreateEmptyResults(nationalCodes);
}
catch (Exception ex)
{
_logger.LogError(ex, "Unexpected error calling Daya API");
return CreateEmptyResults(nationalCodes);
}
}
/// <summary>
/// تبدیل پاسخ API دایا به مدل داخلی
/// </summary>
private List<DayaLoanStatusResult> MapApiResponseToResults(
List<DayaContractData> apiData,
List<string> requestedNationalCodes)
{
var results = new List<DayaLoanStatusResult>();
// برای هر کد ملی درخواستی
foreach (var nationalCode in requestedNationalCodes)
{
// پیدا کردن آخرین قرارداد این کد ملی (براساس تاریخ)
var latestContract = apiData
.Where(d => d.NationalCode == nationalCode)
.OrderByDescending(d => d.DateTime)
.FirstOrDefault();
if (latestContract != null)
{
// تعیین وضعیت براساس StatusDescription
var status = MapStatusDescription(latestContract.StatusDescription);
results.Add(new DayaLoanStatusResult
{
NationalCode = nationalCode,
Status = status,
ContractNumber = latestContract.ApplicationNo
});
_logger.LogDebug("Daya contract found for {NationalCode}: {ContractNo}, Status: {Status}",
nationalCode, latestContract.ApplicationNo, latestContract.StatusDescription);
}
else
{
// هیچ قراردادی برای این کد ملی پیدا نشد
results.Add(new DayaLoanStatusResult
{
NationalCode = nationalCode,
Status = DayaLoanStatus.NotRequested,
ContractNumber = null
});
}
}
return results;
}
/// <summary>
/// تبدیل StatusDescription دایا به Enum داخلی
/// </summary>
private DayaLoanStatus MapStatusDescription(string statusDescription)
{
if (string.IsNullOrWhiteSpace(statusDescription))
return DayaLoanStatus.NotRequested;
// براساس مستند دایا: "فعال شده (در انتظار تسویه)" = وام تایید شده
if (statusDescription.Contains("فعال شده") || statusDescription.Contains("در انتظار تسویه"))
return DayaLoanStatus.PendingReceive;
if (statusDescription.Contains("رد شده") || statusDescription.Contains("رد"))
return DayaLoanStatus.Rejected;
if (statusDescription.Contains("بررسی") || statusDescription.Contains("در حال"))
return DayaLoanStatus.UnderReview;
// پیش‌فرض
return DayaLoanStatus.NotRequested;
}
/// <summary>
/// ساخت نتایج خالی در صورت خطا
/// </summary>
private List<DayaLoanStatusResult> CreateEmptyResults(List<string> nationalCodes)
{
return nationalCodes.Select(nc => new DayaLoanStatusResult
{
NationalCode = nc,
Status = DayaLoanStatus.NotRequested,
ContractNumber = null
}).ToList();
}
}
#region Daya API Models
/// <summary>
/// Request body برای /api/merchant/contracts
/// </summary>
internal class DayaContractsRequest
{
[JsonPropertyName("NationalCodes")]
public List<string> NationalCodes { get; set; } = new();
}
/// <summary>
/// Response از /api/merchant/contracts
/// </summary>
internal class DayaContractsResponse
{
[JsonPropertyName("succeed")]
public bool Succeed { get; set; }
[JsonPropertyName("code")]
public int Code { get; set; }
[JsonPropertyName("message")]
public string? Message { get; set; }
[JsonPropertyName("data")]
public List<DayaContractData>? Data { get; set; }
}
/// <summary>
/// داده‌های قرارداد در پاسخ API
/// </summary>
internal class DayaContractData
{
[JsonPropertyName("nationalCode")]
public string NationalCode { get; set; } = string.Empty;
[JsonPropertyName("applicationNo")]
public string ApplicationNo { get; set; } = string.Empty;
[JsonPropertyName("statusDescription")]
public string StatusDescription { get; set; } = string.Empty;
[JsonPropertyName("dateTime")]
public DateTime DateTime { get; set; }
}
#endregion

View File

@@ -211,7 +211,7 @@ public class DayaPaymentService : IPaymentGatewayService
{
IsSuccess = false,
Message = "فرمت شماره شبا نامعتبر است",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
@@ -222,7 +222,7 @@ public class DayaPaymentService : IPaymentGatewayService
{
IsSuccess = false,
Message = "حداقل مبلغ برداشت 10,000 تومان است",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
@@ -252,7 +252,7 @@ public class DayaPaymentService : IPaymentGatewayService
{
IsSuccess = false,
Message = $"خطا در واریز: {response.StatusCode}",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
@@ -264,7 +264,7 @@ public class DayaPaymentService : IPaymentGatewayService
{
IsSuccess = false,
Message = "پاسخ نامعتبر از درگاه",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
@@ -278,7 +278,7 @@ public class DayaPaymentService : IPaymentGatewayService
BankRefId = result.BankRefId,
TrackingCode = result.TrackingCode,
Message = result.Message,
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
catch (Exception ex)
@@ -288,7 +288,7 @@ public class DayaPaymentService : IPaymentGatewayService
{
IsSuccess = false,
Message = "خطا در پردازش واریز",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
}

View File

@@ -91,7 +91,7 @@ public class MockPaymentGatewayService : IPaymentGatewayService
{
IsSuccess = false,
Message = "فرمت شماره شبا نامعتبر است",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
@@ -103,7 +103,7 @@ public class MockPaymentGatewayService : IPaymentGatewayService
{
IsSuccess = false,
Message = "حداقل مبلغ برداشت 10,000 تومان است",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
@@ -121,7 +121,7 @@ public class MockPaymentGatewayService : IPaymentGatewayService
BankRefId = bankRefId,
TrackingCode = trackingCode,
Message = $"واریز {request.Amount:N0} تومان به حساب {request.Iban} با موفقیت انجام شد (Mock)",
ProcessedAt = DateTime.UtcNow
ProcessedAt = DateTime.Now
};
}
}

View File

@@ -3,7 +3,7 @@
<TargetFramework>net9.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<Version>0.0.142</Version>
<Version>0.0.149</Version>
<DebugType>None</DebugType>
<DebugSymbols>False</DebugSymbols>
<GeneratePackageOnBuild>False</GeneratePackageOnBuild>
@@ -60,7 +60,7 @@
<Target Name="PushToFoursatNuget" AfterTargets="Pack">
<PropertyGroup>
<NugetPackagePath>$(PackageOutputPath)$(PackageId).$(Version).nupkg</NugetPackagePath>
<PushCommand>dotnet nuget push **/*.nupkg --source https://git.afrino.co/api/packages/FourSat/nuget/index.json --api-key 061a5cb15517c6da39c16cfce8556c55ae104d0d --skip-duplicate &amp;&amp; del "$(NugetPackagePath)"</PushCommand>
<PushCommand>dotnet nuget push **/*.nupkg --source https://git.afrino.co/api/packages/FourSat/nuget/index.json --api-key 061a5cb15517c6da39c16cfce8556c55ae104d0d --skip-duplicate </PushCommand>
</PropertyGroup>
<Exec Command="$(PushCommand)" />

View File

@@ -50,6 +50,19 @@ service ClubMembershipContract
get: "/ClubMembership/GetStatistics"
};
};
// New: Admin Club Feature Management
rpc GetUserClubFeatures(GetUserClubFeaturesRequest) returns (GetUserClubFeaturesResponse){
option (google.api.http) = {
get: "/ClubFeature/GetUserFeatures"
};
};
rpc ToggleUserClubFeature(ToggleUserClubFeatureRequest) returns (ToggleUserClubFeatureResponse){
option (google.api.http) = {
post: "/ClubFeature/ToggleFeature"
body: "*"
};
};
}
// Activate Command
@@ -205,3 +218,43 @@ message MonthlyMembershipTrend
int32 expirations = 3;
int32 net_change = 4;
}
// GetUserClubFeatures Query
message GetUserClubFeaturesRequest
{
int64 user_id = 1;
}
message GetUserClubFeaturesResponse
{
repeated UserClubFeatureModel features = 1;
}
message UserClubFeatureModel
{
int64 id = 1;
int64 user_id = 2;
int64 club_membership_id = 3;
int64 club_feature_id = 4;
string feature_title = 5;
string feature_description = 6;
bool is_active = 7;
google.protobuf.Timestamp granted_at = 8;
string notes = 9;
}
// ToggleUserClubFeature Command
message ToggleUserClubFeatureRequest
{
int64 user_id = 1;
int64 club_feature_id = 2;
bool is_active = 3;
}
message ToggleUserClubFeatureResponse
{
bool success = 1;
string message = 2;
google.protobuf.Int64Value user_club_feature_id = 3;
google.protobuf.BoolValue is_active = 4;
}

View File

@@ -105,6 +105,11 @@ service CommissionContract
get: "/Commission/GetWorkerLogs"
};
};
rpc GetAvailableWeeks(GetAvailableWeeksRequest) returns (GetAvailableWeeksResponse){
option (google.api.http) = {
get: "/Commission/GetAvailableWeeks"
};
};
// Financial Reports
rpc GetWithdrawalReports(GetWithdrawalReportsRequest) returns (GetWithdrawalReportsResponse){
@@ -247,9 +252,39 @@ message CommissionPayoutHistoryModel
int32 action = 9; // CommissionPayoutAction enum
string performed_by = 10;
string reason = 11;
google.protobuf.Timestamp created = 12;
google.protobuf.Timestamp created = 13;
}
// ============ GetAvailableWeeks ============
message GetAvailableWeeksRequest
{
int32 future_weeks_count = 1; // تعداد هفته‌های آینده (پیش‌فرض: 4)
int32 past_weeks_count = 2; // تعداد هفته‌های گذشته (پیش‌فرض: 12)
}
message GetAvailableWeeksResponse
{
WeekInfo current_week = 1;
repeated WeekInfo calculated_weeks = 2;
repeated WeekInfo pending_weeks = 3;
repeated WeekInfo future_weeks = 4;
}
message WeekInfo
{
string week_number = 1; // YYYY-Www format
google.protobuf.Timestamp start_date = 2;
google.protobuf.Timestamp end_date = 3;
bool is_calculated = 4;
google.protobuf.Timestamp calculated_at = 5;
string last_execution_status = 6;
int64 total_pool_amount = 7;
int32 eligible_users_count = 8;
string display_text = 9; // نمایش فارسی برای UI
}
// GetUserWeeklyBalances Query
message GetUserWeeklyBalancesRequest
{

View File

@@ -38,6 +38,13 @@ service ManualPaymentContract
get: "/GetAllManualPayments"
};
};
rpc ProcessManualMembershipPayment(ProcessManualMembershipPaymentRequest) returns (ProcessManualMembershipPaymentResponse){
option (google.api.http) = {
post: "/ProcessManualMembershipPayment"
body: "*"
};
};
}
// Enums mirroring CMSMicroservice.Domain.Enums.ManualPaymentType
@@ -128,3 +135,19 @@ message ManualPaymentModel
google.protobuf.Timestamp created = 19;
}
message ProcessManualMembershipPaymentRequest
{
int64 user_id = 1;
int64 amount = 2;
string reference_number = 3;
google.protobuf.StringValue description = 4;
}
message ProcessManualMembershipPaymentResponse
{
int64 transaction_id = 1;
int64 order_id = 2;
int64 new_wallet_balance = 3;
string message = 4;
}

View File

@@ -85,26 +85,74 @@ message GetUserNetworkRequest
message GetUserNetworkResponse
{
// اطلاعات اصلی کاربر
int64 id = 1;
int64 user_id = 2;
string user_name = 3;
google.protobuf.Int64Value parent_id = 4;
string parent_name = 5;
int32 network_leg = 6; // NetworkLeg enum
google.protobuf.Int64Value left_child_id = 7;
string left_child_name = 8;
google.protobuf.Int64Value right_child_id = 9;
string right_child_name = 10;
int32 network_level = 11;
string referral_code = 12;
google.protobuf.Timestamp joined_at = 13;
google.protobuf.Timestamp created = 14;
string mobile = 4;
string email = 5;
string national_code = 6;
string referral_code = 7;
bool is_mobile_verified = 8;
google.protobuf.Timestamp birth_date = 9;
google.protobuf.Timestamp joined_at = 10;
// اطلاعات والد
google.protobuf.Int64Value parent_id = 11;
string parent_name = 12;
string parent_mobile = 13;
// موقعیت در شبکه
int32 network_leg = 14; // NetworkLeg enum
int32 network_level = 15;
bool is_in_network = 16;
// اطلاعات فرزند چپ
google.protobuf.Int64Value left_child_id = 17;
string left_child_name = 18;
string left_child_mobile = 19;
google.protobuf.Timestamp left_child_joined_at = 20;
// اطلاعات فرزند راست
google.protobuf.Int64Value right_child_id = 21;
string right_child_name = 22;
string right_child_mobile = 23;
google.protobuf.Timestamp right_child_joined_at = 24;
// آمار فرزندان مستقیم
int32 total_children = 25;
int32 left_child_count = 26;
int32 right_child_count = 27;
// آمار کل شبکه
int32 total_left_leg_members = 28;
int32 total_right_leg_members = 29;
int32 total_network_size = 30;
int32 max_network_depth = 31;
// اطلاعات پکیج و دایا
bool has_received_daya_credit = 32;
google.protobuf.Timestamp daya_credit_received_at = 33;
int32 package_purchase_method = 34;
bool has_purchased_golden_package = 35;
// آمار مالی
double total_earned_commission = 36;
double total_paid_commission = 37;
double pending_commission = 38;
int32 total_balances_earned = 39;
// آمار فعالیت
int32 active_members_in_network = 40;
int32 inactive_members_in_network = 41;
google.protobuf.Timestamp created = 42;
}
// GetNetworkTree Query
message GetNetworkTreeRequest
{
int64 root_user_id = 1;
int64 user_id = 1;
google.protobuf.Int32Value max_depth = 2;
google.protobuf.BoolValue only_active = 3;
}

View File

@@ -129,20 +129,21 @@ message GetAllUserByFilterRequest
}
message GetAllUserByFilterFilter
{
google.protobuf.Int64Value id = 1;
google.protobuf.StringValue first_name = 2;
google.protobuf.StringValue last_name = 3;
google.protobuf.StringValue mobile = 4;
google.protobuf.StringValue national_code = 5;
google.protobuf.StringValue avatar_path = 6;
google.protobuf.Int64Value parent_id = 7;
google.protobuf.StringValue referral_code = 8;
google.protobuf.BoolValue is_mobile_verified = 9;
google.protobuf.Timestamp mobile_verified_at = 10;
google.protobuf.BoolValue email_notifications = 11;
google.protobuf.BoolValue sms_notifications = 12;
google.protobuf.BoolValue push_notifications = 13;
google.protobuf.Timestamp birth_date = 14;
google.protobuf.StringValue search_text = 1;
google.protobuf.Int64Value id = 2;
google.protobuf.StringValue first_name = 3;
google.protobuf.StringValue last_name = 4;
google.protobuf.StringValue mobile = 5;
google.protobuf.StringValue national_code = 6;
google.protobuf.StringValue avatar_path = 7;
google.protobuf.Int64Value parent_id = 8;
google.protobuf.StringValue referral_code = 9;
google.protobuf.BoolValue is_mobile_verified = 10;
google.protobuf.Timestamp mobile_verified_at = 11;
google.protobuf.BoolValue email_notifications = 12;
google.protobuf.BoolValue sms_notifications = 13;
google.protobuf.BoolValue push_notifications = 14;
google.protobuf.Timestamp birth_date = 15;
}
message GetAllUserByFilterResponse
{

View File

@@ -14,6 +14,7 @@
<PackageReference Include="Grpc.Net.Client" Version="2.54.0" />
<PackageReference Include="Hangfire.AspNetCore" Version="1.8.22" />
<PackageReference Include="Hangfire.SqlServer" Version="1.8.22" />
<PackageReference Include="Mapster" Version="7.4.0" />
<PackageReference Include="Mapster.DependencyInjection" Version="1.0.0" />
<PackageReference Include="MediatR" Version="11.0.0" />

View File

@@ -0,0 +1,47 @@
using CMSMicroservice.Application.ClubFeatureCQ.Commands.ToggleUserClubFeature;
using CMSMicroservice.Application.ClubFeatureCQ.Queries.GetUserClubFeatures;
using CMSMicroservice.Protobuf.Protos.ClubMembership;
using Google.Protobuf.WellKnownTypes;
using System;
using System.Collections.Generic;
namespace CMSMicroservice.WebApi.Common.Mappings;
public class ClubFeatureProfile : IRegister
{
void IRegister.Register(TypeAdapterConfig config)
{
// GetUserClubFeaturesRequest → GetUserClubFeaturesQuery
config.NewConfig<GetUserClubFeaturesRequest, GetUserClubFeaturesQuery>()
.Map(dest => dest.UserId, src => src.UserId);
// UserClubFeatureDto → UserClubFeatureModel
config.NewConfig<UserClubFeatureDto, UserClubFeatureModel>()
.Map(dest => dest.Id, src => src.Id)
.Map(dest => dest.UserId, src => src.UserId)
.Map(dest => dest.ClubMembershipId, src => src.ClubMembershipId)
.Map(dest => dest.ClubFeatureId, src => src.ClubFeatureId)
.Map(dest => dest.FeatureTitle, src => src.FeatureTitle)
.Map(dest => dest.FeatureDescription, src => src.FeatureDescription ?? "")
.Map(dest => dest.IsActive, src => src.IsActive)
.Map(dest => dest.GrantedAt, src => Timestamp.FromDateTime(DateTime.SpecifyKind(src.GrantedAt, DateTimeKind.Utc)))
.Map(dest => dest.Notes, src => src.Notes ?? "");
// List<UserClubFeatureDto> → GetUserClubFeaturesResponse
config.NewConfig<List<UserClubFeatureDto>, GetUserClubFeaturesResponse>()
.Map(dest => dest.Features, src => src.Adapt<List<UserClubFeatureModel>>());
// ToggleUserClubFeatureRequest → ToggleUserClubFeatureCommand
config.NewConfig<ToggleUserClubFeatureRequest, ToggleUserClubFeatureCommand>()
.Map(dest => dest.UserId, src => src.UserId)
.Map(dest => dest.ClubFeatureId, src => src.ClubFeatureId)
.Map(dest => dest.IsActive, src => src.IsActive);
// ToggleUserClubFeatureResponse (App) → ToggleUserClubFeatureResponse (Proto)
config.NewConfig<Application.ClubFeatureCQ.Commands.ToggleUserClubFeature.ToggleUserClubFeatureResponse, Protobuf.Protos.ClubMembership.ToggleUserClubFeatureResponse>()
.Map(dest => dest.Success, src => src.Success)
.Map(dest => dest.Message, src => src.Message)
.Map(dest => dest.UserClubFeatureId, src => src.UserClubFeatureId.HasValue ? (long?)src.UserClubFeatureId.Value : null)
.Map(dest => dest.IsActive, src => src.IsActive.HasValue ? (bool?)src.IsActive.Value : null);
}
}

View File

@@ -0,0 +1,39 @@
using CMSMicroservice.Application.CommissionCQ.Queries.GetAvailableWeeks;
using CMSMicroservice.Protobuf.Protos.Commission;
using Google.Protobuf.WellKnownTypes;
using Mapster;
namespace CMSMicroservice.WebApi.Common.Mappings;
public class CommissionProfile : IRegister
{
public void Register(TypeAdapterConfig config)
{
// GetAvailableWeeks Request Mapping
config.NewConfig<GetAvailableWeeksRequest, GetAvailableWeeksQuery>()
.Map(dest => dest.FutureWeeksCount, src => src.FutureWeeksCount > 0 ? src.FutureWeeksCount : 4)
.Map(dest => dest.PastWeeksCount, src => src.PastWeeksCount > 0 ? src.PastWeeksCount : 12);
// GetAvailableWeeks Response Mapping
config.NewConfig<GetAvailableWeeksResponseDto, GetAvailableWeeksResponse>()
.Map(dest => dest.CurrentWeek, src => src.CurrentWeek)
.Map(dest => dest.CalculatedWeeks, src => src.CalculatedWeeks)
.Map(dest => dest.PendingWeeks, src => src.PendingWeeks)
.Map(dest => dest.FutureWeeks, src => src.FutureWeeks);
// WeekInfo Mapping
config.NewConfig<WeekInfoDto, WeekInfo>()
.Map(dest => dest.WeekNumber, src => src.WeekNumber)
.Map(dest => dest.StartDate, src => Timestamp.FromDateTime(src.StartDate.ToUniversalTime()))
.Map(dest => dest.EndDate, src => Timestamp.FromDateTime(src.EndDate.ToUniversalTime()))
.Map(dest => dest.IsCalculated, src => src.IsCalculated)
.Map(dest => dest.CalculatedAt, src => src.CalculatedAt.HasValue
? Timestamp.FromDateTime(src.CalculatedAt.Value.ToUniversalTime())
: null)
.Map(dest => dest.LastExecutionStatus, src => src.LastExecutionStatus ?? string.Empty)
.Map(dest => dest.TotalPoolAmount, src => src.TotalPoolAmount ?? 0)
.Map(dest => dest.EligibleUsersCount, src => src.EligibleUsersCount ?? 0)
.Map(dest => dest.DisplayText, src => src.DisplayText);
}
}

View File

@@ -0,0 +1,105 @@
using CMSMicroservice.Application.NetworkMembershipCQ.Queries.GetNetworkTree;
using CMSMicroservice.Application.NetworkMembershipCQ.Queries.GetUserNetworkPosition;
using CMSMicroservice.Protobuf.Protos.NetworkMembership;
using CMSMicroservice.Domain.Enums;
namespace CMSMicroservice.WebApi.Common.Mappings;
public class NetworkMembershipProfile : IRegister
{
void IRegister.Register(TypeAdapterConfig config)
{
// Request mapping
config.NewConfig<GetNetworkTreeRequest, GetNetworkTreeQuery>()
.Map(dest => dest.UserId, src => src.UserId)
.Map(dest => dest.MaxDepth, src => src.MaxDepth != null && src.MaxDepth.Value > 0 ? src.MaxDepth.Value : 3);
// Response mapping: تبدیل درخت به لیست مسطح
config.NewConfig<NetworkTreeDto, GetNetworkTreeResponse>()
.MapWith(src => ConvertTreeToResponse(src));
config.NewConfig<UserNetworkPositionDto, GetUserNetworkResponse>()
.Map(dest => dest.Id, src => src.UserId)
.Map(dest => dest.UserId, src => src.UserId)
.Map(dest => dest.UserName, src => (src.FirstName + " " + src.LastName).Trim())
.Map(dest => dest.Mobile, src => src.Mobile ?? "")
.Map(dest => dest.Email, src => src.Email ?? "")
.Map(dest => dest.NationalCode, src => src.NationalCode ?? "")
.Map(dest => dest.ReferralCode, src => src.ReferralCode)
.Map(dest => dest.IsMobileVerified, src => src.IsMobileVerified)
.Map(dest => dest.ParentId, src => src.NetworkParentId)
.Map(dest => dest.ParentName, src => src.ParentFullName ?? "")
.Map(dest => dest.ParentMobile, src => src.ParentMobile ?? "")
.Map(dest => dest.NetworkLeg, src => (int)(src.LegPosition ?? NetworkLeg.Left))
.Map(dest => dest.NetworkLevel, src => 0) // Deprecated field
.Map(dest => dest.IsInNetwork, src => src.IsInNetwork)
.Map(dest => dest.LeftChildId, src => src.LeftChildId)
.Map(dest => dest.LeftChildName, src => src.LeftChildFullName ?? "")
.Map(dest => dest.LeftChildMobile, src => src.LeftChildMobile ?? "")
.Map(dest => dest.RightChildId, src => src.RightChildId)
.Map(dest => dest.RightChildName, src => src.RightChildFullName ?? "")
.Map(dest => dest.RightChildMobile, src => src.RightChildMobile ?? "")
.Map(dest => dest.TotalChildren, src => src.TotalChildren)
.Map(dest => dest.LeftChildCount, src => src.LeftChildCount)
.Map(dest => dest.RightChildCount, src => src.RightChildCount)
.Map(dest => dest.TotalLeftLegMembers, src => src.TotalLeftLegMembers)
.Map(dest => dest.TotalRightLegMembers, src => src.TotalRightLegMembers)
.Map(dest => dest.TotalNetworkSize, src => src.TotalNetworkSize)
.Map(dest => dest.MaxNetworkDepth, src => src.MaxNetworkDepth)
.Map(dest => dest.HasReceivedDayaCredit, src => src.HasReceivedDayaCredit)
.Map(dest => dest.PackagePurchaseMethod, src => (int)src.PackagePurchaseMethod)
.Map(dest => dest.HasPurchasedGoldenPackage, src => src.HasPurchasedGoldenPackage)
.Map(dest => dest.TotalEarnedCommission, src => (double)src.TotalEarnedCommission)
.Map(dest => dest.TotalPaidCommission, src => (double)src.TotalPaidCommission)
.Map(dest => dest.PendingCommission, src => (double)src.PendingCommission)
.Map(dest => dest.TotalBalancesEarned, src => src.TotalBalancesEarned)
.Map(dest => dest.ActiveMembersInNetwork, src => src.ActiveMembersInNetwork)
.Map(dest => dest.InactiveMembersInNetwork, src => src.InactiveMembersInNetwork);
}
private static GetNetworkTreeResponse ConvertTreeToResponse(NetworkTreeDto? treeDto)
{
var response = new GetNetworkTreeResponse();
if (treeDto == null)
{
return response;
}
// تبدیل درخت به لیست مسطح
FlattenTree(treeDto, response.Nodes, null);
return response;
}
private static void FlattenTree(NetworkTreeDto node, Google.Protobuf.Collections.RepeatedField<NetworkTreeNodeModel> nodesList, long? parentId)
{
var protoNode = new NetworkTreeNodeModel
{
UserId = node.UserId,
UserName = $"{node.FirstName} {node.LastName}".Trim(),
NetworkLeg = (int)(node.LegPosition ?? NetworkLeg.Left),
NetworkLevel = node.CurrentDepth,
IsActive = true
};
if (parentId.HasValue)
{
protoNode.ParentId = parentId.Value;
}
nodesList.Add(protoNode);
// بازگشتی برای فرزندان
if (node.LeftChild != null)
{
FlattenTree(node.LeftChild, nodesList, node.UserId);
}
if (node.RightChild != null)
{
FlattenTree(node.RightChild, nodesList, node.UserId);
}
}
}

View File

@@ -38,9 +38,9 @@ public class AdminController : ControllerBase
{
_logger.LogInformation("🔧 Manual trigger requested by admin for week: {WeekNumber}", weekNumber ?? "previous");
// Enqueue immediate job execution
// Enqueue immediate job execution with specified week number
var jobId = _backgroundJobClient.Enqueue<WeeklyCommissionJob>(
job => job.ExecuteAsync(CancellationToken.None));
job => job.ExecuteAsync(weekNumber, CancellationToken.None));
_logger.LogInformation("✅ Job enqueued with ID: {JobId}", jobId);

View File

@@ -13,7 +13,7 @@ COPY ["CMSMicroservice.Application/CMSMicroservice.Application.csproj", "CMSMicr
COPY ["CMSMicroservice.Domain/CMSMicroservice.Domain.csproj", "CMSMicroservice.Domain/"]
COPY ["CMSMicroservice.Infrastructure/CMSMicroservice.Infrastructure.csproj", "CMSMicroservice.Infrastructure/"]
COPY ["CMSMicroservice.Protobuf/CMSMicroservice.Protobuf.csproj", "CMSMicroservice.Protobuf/"]
RUN dotnet restore "CMSMicroservice.WebApi/CMSMicroservice.WebApi.csproj"
RUN dotnet restore "CMSMicroservice.WebApi/CMSMicroservice.WebApi.csproj" --configfile ../NuGet.config
COPY . .
WORKDIR "/src/CMSMicroservice.WebApi"
RUN dotnet build "CMSMicroservice.WebApi.csproj" -c Release -o /app/build

Some files were not shown because too many files have changed in this diff Show More