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>
146 lines
5.0 KiB
Dart
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,
|
|
);
|
|
}
|
|
}
|