summaryrefslogtreecommitdiff
path: root/src/main/java/events/internal/VSProtocolEvent.java
blob: 41adc3c75340636aa720658fe188c21487ffe08d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
package events.internal;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.HashMap;
import java.util.Map;

import core.VSInternalProcess;
import core.VSTask;
import core.VSTaskManager;
import events.VSAbstractEvent;
import events.VSCopyableEvent;
import events.VSRegisteredEvents;
import protocols.VSAbstractProtocol;
import serialize.VSSerialize;

/**
 * The class VSProtocolEvent, this event is used if a protocol (server or
 * client part) of a process gets enabled or disabled, an object of this class
 * can be for 4 different purporses! Activation of the client protocol,
 * deactivation of the client protocol, activation of the server protocol,
 * deactivation of the server protocol.
 *
 * @author Paul C. Buetow
 */
public class VSProtocolEvent extends VSAbstractInternalEvent
    implements VSCopyableEvent {
    /** The protocol classname. */
    private String protocolClassname;

    /** The event is a client protocol if true. Else it is a server protocol */
    private boolean isClientProtocol;
    
    @Override
    public boolean isProtocolEvent() {
        return true;
    }
    
    @Override
    public int getEventPriority() {
        return PRIORITY_MEDIUM;
    }

    /** The event is a protocol activation if true. Else it is a deactivation */
    private boolean isProtocolActivation;

    /** Optional long preference overrides applied when the protocol starts. */
    private HashMap<String, Long> longOverrides;

    /* (non-Javadoc)
     * @see events.VSCopyableEvent#initCopy(events.VSAbstractEvent)
     */
    public void initCopy(VSAbstractEvent copy) {
        VSProtocolEvent protocolEventCopy = (VSProtocolEvent) copy;
        protocolEventCopy.isClientProtocol(isClientProtocol);
        protocolEventCopy.isProtocolActivation(isProtocolActivation);
        protocolEventCopy.setProtocolClassname(protocolClassname);
        if (longOverrides != null) {
            protocolEventCopy.longOverrides =
                new HashMap<String, Long>(longOverrides);
        }
    }

    /* (non-Javadoc)
     * @see events.VSAbstractEvent#onInit()
     */
    public void onInit() {
        setClassname(getClass().toString());
    }

    /**
     * Sets if it is a client protocol activation/deactivation.
     *
     * @param isClientProtocol the event is client protocol if true. the event
     * is a server protocol if false.
     */
    public void isClientProtocol(boolean isClientProtocol) {
        this.isClientProtocol = isClientProtocol;
    }

    /**
     * Checks if it is a client protocol activation/deactivation.
     *
     * @return the event is client protocol if true. the event
     * is a server protocol if false.
     */
    public boolean isClientProtocol() {
        return isClientProtocol;
    }

    /**
     * Sets if it is protocol activation.
     *
     * @param isProtocolActivation true, if it is a protocol activation. false,
     *	if it is a protocol deactivation.
     */
    public void isProtocolActivation(boolean isProtocolActivation) {
        this.isProtocolActivation = isProtocolActivation;
    }

    /**
     * Checks if it is protocol activation.
     *
     * @return true, if it is a protocol activation. false, if it is a protocol
     *	deactivation.
     */
    public boolean isProtocolActivation() {
        return isProtocolActivation;
    }

    /**
     * Sets the protocol classname.
     *
     * @param protocolClassname the new protocol classname
     */
    public void setProtocolClassname(String protocolClassname) {
        this.protocolClassname = protocolClassname;
    }

    /**
     * Overrides a long preference before the protocol starts.
     *
     * @param key the preference key
     * @param value the value to apply
     */
    public void setLongOverride(String key, long value) {
        if (longOverrides == null) {
            longOverrides = new HashMap<String, Long>();
        }

        longOverrides.put(key, Long.valueOf(value));
    }
    
    /**
     * Checks if we should schedule a protocol start task.
     * We should NOT schedule if the protocol is already scheduled at the current time.
     * This prevents duplicate execution when loading saved simulations.
     *
     * @param process the process to check
     * @param protocol the protocol to check for
     * @return true if we should schedule, false if already scheduled
     */
    private boolean shouldScheduleProtocolStart(VSInternalProcess process, VSAbstractProtocol protocol) {
        // Check process-local tasks
        for (VSTask task : process.getTasks()) {
            if (task.getEvent() == protocol && task.getTaskTime() == process.getTime()) {
                // Protocol is already scheduled at this time
                return false;
            }
        }
        
        // Check global tasks  
        VSTaskManager taskManager = process.getSimulatorCanvas().getTaskManager();
        try {
            // Use reflection to access global tasks
            java.lang.reflect.Field globalTasksField = VSTaskManager.class.getDeclaredField("globalTasks");
            globalTasksField.setAccessible(true);
            @SuppressWarnings("unchecked")
            java.util.Queue<VSTask> globalTasks = (java.util.Queue<VSTask>) globalTasksField.get(taskManager);
            
            for (VSTask task : globalTasks) {
                if (task.getEvent() == protocol && task.getTaskTime() == process.getGlobalTime()) {
                    // Protocol is already scheduled at this time
                    return false;
                }
            }
        } catch (Exception e) {
            // If we can't check, assume we should schedule
            System.err.println("Warning: Could not check for duplicate protocol tasks: " + e.getMessage());
        }
        
        return true;
    }

    /* (non-Javadoc)
     * @see events.VSAbstractEvent#onStart()
     */
    public void onStart() {
        VSInternalProcess internalProcess = (VSInternalProcess) process;
        VSAbstractProtocol protocol =
            internalProcess.getProtocolObject(protocolClassname);

        applyLongOverrides(protocol);

        if (isClientProtocol)
            protocol.isClient(isProtocolActivation);
        else
            protocol.isServer(isProtocolActivation);

        StringBuffer buffer = new StringBuffer();
        buffer.append(VSRegisteredEvents.getShortnameByClassname(protocolClassname));
        if (buffer.length() == 0) {
            buffer.append(protocolClassname);
        }

        buffer.append(" ");
        buffer.append(isClientProtocol
                      ? prefs.getString("lang.client")
                      : prefs.getString("lang.server"));

        buffer.append(" ");
        buffer.append(isProtocolActivation
                      ? prefs.getString("lang.activated")
                      : prefs.getString("lang.deactivated"));

        log(buffer.toString());
        
        // If this is an activation, schedule the protocol to start immediately
        // This ensures that protocols with HAS_ON_SERVER_START or HAS_ON_CLIENT_START
        // will have their onServerStart() or onClientStart() methods called
        // 
        // However, we should NOT schedule if the protocol is already scheduled to run.
        // This can happen when loading from a saved simulation where both the activation
        // event and the resulting protocol task were saved.
        if (isProtocolActivation && shouldScheduleProtocolStart(internalProcess, protocol)) {
            // Create a task to start the protocol at the current time
            VSTask startTask = new VSTask(internalProcess.getTime(), 
                                         internalProcess, 
                                         protocol, 
                                         VSTask.LOCAL);
            internalProcess.getSimulatorCanvas().getTaskManager().addTask(startTask);
        }
    }

    /* (non-Javadoc)
     * @see serialize.VSSerializable#serialize(serialize.VSSerialize,
     *	java.io.ObjectOutputStream)
     */
    public synchronized void serialize(VSSerialize serialize,
                                       ObjectOutputStream objectOutputStream)
    throws IOException {
        super.serialize(serialize, objectOutputStream);

        /** For later backwards compatibility, to add more stuff */
        objectOutputStream.writeObject(Boolean.valueOf(false));

        objectOutputStream.writeObject(protocolClassname);
        objectOutputStream.writeObject(Boolean.valueOf(isClientProtocol));
        objectOutputStream.writeObject(Boolean.valueOf(isProtocolActivation));

        /** For later backwards compatibility, to add more stuff */
        objectOutputStream.writeObject(longOverrides == null
                                      ? Boolean.valueOf(false)
                                      : new HashMap<String, Long>(longOverrides));
    }

    /* (non-Javadoc)
     * @see serialize.VSSerializable#deserialize(serialize.VSSerialize,
     *	java.io.ObjectInputStream)
     */
    public synchronized void deserialize(VSSerialize serialize,
                                         ObjectInputStream objectInputStream)
    throws IOException, ClassNotFoundException {
        super.deserialize(serialize, objectInputStream);

        if (VSSerialize.DEBUG)
            System.out.println("Deserializing: VSProtocolEvent");

        /** For later backwards compatibility, to add more stuff */
        objectInputStream.readObject();

        // Set protocolClassname before calling createShortname
        protocolClassname = (String) objectInputStream.readObject();
        isClientProtocol = ((Boolean) objectInputStream.readObject()).booleanValue();
        isProtocolActivation = ((Boolean) objectInputStream.readObject()).booleanValue();

        // Set the event shortname using current localization
        this.setShortname(createShortname(null));

        /** For later backwards compatibility, to add more stuff */
        Object overrides = objectInputStream.readObject();
        if (overrides instanceof Map<?, ?> map) {
            longOverrides = new HashMap<String, Long>();
            for (Map.Entry<?, ?> entry : map.entrySet()) {
                if (entry.getKey() instanceof String
                        && entry.getValue() instanceof Long) {
                    longOverrides.put((String) entry.getKey(),
                                      (Long) entry.getValue());
                }
            }
        }
    }

    protected String createShortname(String savedShortname) {
        // Handle case where this is called during parent deserialization
        // before our fields are initialized
        if (protocolClassname == null || prefs == null) {
            return savedShortname != null ? savedShortname : "Protocol Event";
        }
        
        // Always use current localization strings
        String protocolShortname = VSRegisteredEvents.getShortnameByClassname(protocolClassname);
        if (protocolShortname == null) {
            protocolShortname = protocolClassname;
        }
        
        String clientServer = isClientProtocol ? 
            prefs.getString("lang.client") : 
            prefs.getString("lang.server");
        String activateDeactivate = isProtocolActivation ?
            prefs.getString("lang.activated") :
            prefs.getString("lang.deactivated");
        return protocolShortname + " " + clientServer + " " + activateDeactivate;
    }

    private void applyLongOverrides(VSAbstractProtocol protocol) {
        if (longOverrides == null || longOverrides.isEmpty()) {
            return;
        }

        for (Map.Entry<String, Long> entry : longOverrides.entrySet()) {
            protocol.setLong(entry.getKey(), entry.getValue().longValue());
        }
    }
}