Files
geutebruck/geutebruck_app/lib/data/models/action_mapping_model.dart
Administrator 14893e62a5 feat: Geutebruck GeViScope/GeViSoft Action Mapping System - MVP
This MVP release provides a complete full-stack solution for managing action mappings
in Geutebruck's GeViScope and GeViSoft video surveillance systems.

## Features

### Flutter Web Application (Port 8081)
- Modern, responsive UI for managing action mappings
- Action picker dialog with full parameter configuration
- Support for both GSC (GeViScope) and G-Core server actions
- Consistent UI for input and output actions with edit/delete capabilities
- Real-time action mapping creation, editing, and deletion
- Server categorization (GSC: prefix for GeViScope, G-Core: prefix for G-Core servers)

### FastAPI REST Backend (Port 8000)
- RESTful API for action mapping CRUD operations
- Action template service with comprehensive action catalog (247 actions)
- Server management (G-Core and GeViScope servers)
- Configuration tree reading and writing
- JWT authentication with role-based access control
- PostgreSQL database integration

### C# SDK Bridge (gRPC, Port 50051)
- Native integration with GeViSoft SDK (GeViProcAPINET_4_0.dll)
- Action mapping creation with correct binary format
- Support for GSC and G-Core action types
- Proper Camera parameter inclusion in action strings (fixes CrossSwitch bug)
- Action ID lookup table with server-specific action IDs
- Configuration reading/writing via SetupClient

## Bug Fixes
- **CrossSwitch Bug**: GSC and G-Core actions now correctly display camera/PTZ head parameters in GeViSet
- Action strings now include Camera parameter: `@ PanLeft (Comment: "", Camera: 101028)`
- Proper filter flags and VideoInput=0 for action mappings
- Correct action ID assignment (4198 for GSC, 9294 for G-Core PanLeft)

## Technical Stack
- **Frontend**: Flutter Web, Dart, Dio HTTP client
- **Backend**: Python FastAPI, PostgreSQL, Redis
- **SDK Bridge**: C# .NET 8.0, gRPC, GeViSoft SDK
- **Authentication**: JWT tokens
- **Configuration**: GeViSoft .set files (binary format)

## Credentials
- GeViSoft/GeViScope: username=sysadmin, password=masterkey
- Default admin: username=admin, password=admin123

## Deployment
All services run on localhost:
- Flutter Web: http://localhost:8081
- FastAPI: http://localhost:8000
- SDK Bridge gRPC: localhost:50051
- GeViServer: localhost (default port)

Generated with Claude Code (https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-31 18:10:54 +01:00

146 lines
5.0 KiB
Dart

import '../../domain/entities/action_mapping.dart';
import 'action_output.dart';
/// Data model for action mapping with JSON serialization
/// Handles conversion between API (snake_case) and domain entities
class ActionMappingModel {
final String id;
final String name;
final String? description;
final String inputAction;
final Map<String, dynamic> inputParameters;
final List<ActionOutput> outputActions;
final String? geviscopeInstanceScope;
final bool enabled;
final int executionCount;
final DateTime? lastExecuted;
final DateTime createdAt;
final DateTime updatedAt;
final String createdBy;
const ActionMappingModel({
required this.id,
required this.name,
this.description,
required this.inputAction,
this.inputParameters = const {},
required this.outputActions,
this.geviscopeInstanceScope,
this.enabled = true,
this.executionCount = 0,
this.lastExecuted,
required this.createdAt,
required this.updatedAt,
required this.createdBy,
});
/// Create model from JSON (API response with snake_case keys)
factory ActionMappingModel.fromJson(Map<String, dynamic> json) {
// Convert ID from int to string if needed
final id = json['id'].toString();
// Extract input action and parameters from array of objects
final inputActions = json['input_actions'] as List<dynamic>?;
String inputAction;
Map<String, dynamic> inputParameters;
if (inputActions != null && inputActions.isNotEmpty) {
final firstInput = inputActions[0] as Map<String, dynamic>;
inputAction = firstInput['action'] as String;
inputParameters = Map<String, dynamic>.from(firstInput['parameters'] as Map? ?? {});
} else {
inputAction = json['name'] as String; // Fallback to name
inputParameters = {};
}
// Extract output actions and parameters from array of objects
final outputActionsRaw = json['output_actions'] as List<dynamic>?;
final outputActions = outputActionsRaw != null
? outputActionsRaw.map((e) => ActionOutput.fromJson(e as Map<String, dynamic>)).toList()
: <ActionOutput>[];
return ActionMappingModel(
id: id,
name: json['name'] as String,
description: json['description'] as String?,
inputAction: inputAction,
inputParameters: inputParameters,
outputActions: outputActions,
geviscopeInstanceScope: json['geviscope_instance_scope'] as String?,
enabled: json['enabled'] as bool? ?? true,
executionCount: json['execution_count'] as int? ?? 0,
lastExecuted: json['last_executed'] != null
? DateTime.parse(json['last_executed'] as String)
: null,
createdAt: json['created_at'] != null
? DateTime.parse(json['created_at'] as String)
: DateTime.now(),
updatedAt: json['updated_at'] != null
? DateTime.parse(json['updated_at'] as String)
: DateTime.now(),
createdBy: json['created_by'] as String? ?? 'system',
);
}
/// Convert model to JSON (for API requests with snake_case keys)
Map<String, dynamic> toJson() {
return {
'id': id,
'name': name,
if (description != null) 'description': description,
// API expects input_actions as array of objects with parameters
'input_actions': [
{'action': inputAction, 'parameters': inputParameters}
],
// API expects output_actions as array of objects with parameters
'output_actions': outputActions.map((output) => output.toJson()).toList(),
if (geviscopeInstanceScope != null)
'geviscope_instance_scope': geviscopeInstanceScope,
'enabled': enabled,
'execution_count': executionCount,
if (lastExecuted != null) 'last_executed': lastExecuted!.toIso8601String(),
'created_at': createdAt.toIso8601String(),
'updated_at': updatedAt.toIso8601String(),
'created_by': createdBy,
};
}
/// Convert to domain entity
ActionMapping toEntity() {
return ActionMapping(
id: id,
name: name,
description: description,
inputAction: inputAction,
inputParameters: inputParameters,
outputActions: outputActions,
geviscopeInstanceScope: geviscopeInstanceScope,
enabled: enabled,
executionCount: executionCount,
lastExecuted: lastExecuted,
createdAt: createdAt,
updatedAt: updatedAt,
createdBy: createdBy,
);
}
/// Create model from domain entity
factory ActionMappingModel.fromEntity(ActionMapping mapping) {
return ActionMappingModel(
id: mapping.id,
name: mapping.name,
description: mapping.description,
inputAction: mapping.inputAction,
inputParameters: mapping.inputParameters,
outputActions: mapping.outputActions,
geviscopeInstanceScope: mapping.geviscopeInstanceScope,
enabled: mapping.enabled,
executionCount: mapping.executionCount,
lastExecuted: mapping.lastExecuted,
createdAt: mapping.createdAt,
updatedAt: mapping.updatedAt,
createdBy: mapping.createdBy,
);
}
}