Commit | Line | Data |
---|---|---|
12c155f5 | 1 | /******************************************************************************* |
58ffe079 | 2 | * Copyright (c) 2010, 2015 Ericsson, École Polytechnique de Montréal |
ce2388e0 | 3 | * |
12c155f5 FC |
4 | * All rights reserved. This program and the accompanying materials are |
5 | * made available under the terms of the Eclipse Public License v1.0 which | |
6 | * accompanies this distribution, and is available at | |
7 | * http://www.eclipse.org/legal/epl-v10.html | |
ce2388e0 | 8 | * |
12c155f5 FC |
9 | * Contributors: |
10 | * Francois Chouinard - Initial API and implementation | |
b544077e | 11 | * Bernd Hufmann - Added supplementary files handling |
a72a6830 PT |
12 | * Geneviève Bastien - Moved supplementary files handling to parent class, |
13 | * added code to copy trace | |
14 | * Patrick Tasse - Close editors to release resources | |
4962833a JCK |
15 | * Jean-Christian Kouame - added trace properties to be shown into |
16 | * the properties view | |
8f5221c2 | 17 | * Geneviève Bastien - Moved trace type related methods to parent class |
12c155f5 FC |
18 | *******************************************************************************/ |
19 | ||
2bdf0193 | 20 | package org.eclipse.tracecompass.tmf.ui.project.model; |
12c155f5 | 21 | |
5a5c2fc7 | 22 | import java.util.Arrays; |
b75a5a1b | 23 | import java.util.Date; |
12c155f5 | 24 | import java.util.HashMap; |
6e651d8b | 25 | import java.util.LinkedList; |
c068a752 | 26 | import java.util.List; |
12c155f5 FC |
27 | import java.util.Map; |
28 | ||
b75a5a1b PT |
29 | import org.eclipse.core.filesystem.EFS; |
30 | import org.eclipse.core.filesystem.IFileInfo; | |
c4c81d91 | 31 | import org.eclipse.core.resources.IFile; |
5e4bf87d | 32 | import org.eclipse.core.resources.IFolder; |
12c155f5 FC |
33 | import org.eclipse.core.resources.IResource; |
34 | import org.eclipse.core.runtime.CoreException; | |
35 | import org.eclipse.core.runtime.IConfigurationElement; | |
c068a752 | 36 | import org.eclipse.core.runtime.IPath; |
6e651d8b | 37 | import org.eclipse.core.runtime.IProgressMonitor; |
12c155f5 | 38 | import org.eclipse.core.runtime.Platform; |
13f2a21a | 39 | import org.eclipse.core.runtime.URIUtil; |
b75a5a1b | 40 | import org.eclipse.osgi.util.NLS; |
30c7c7d5 | 41 | import org.eclipse.swt.widgets.Display; |
2bdf0193 | 42 | import org.eclipse.tracecompass.internal.tmf.ui.Activator; |
58ffe079 | 43 | import org.eclipse.tracecompass.internal.tmf.ui.editors.ITmfEventsEditorConstants; |
2bdf0193 AM |
44 | import org.eclipse.tracecompass.tmf.core.TmfCommonConstants; |
45 | import org.eclipse.tracecompass.tmf.core.event.ITmfEvent; | |
46 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomTxtEvent; | |
47 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomTxtTrace; | |
48 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomTxtTraceDefinition; | |
49 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomXmlEvent; | |
50 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomXmlTrace; | |
51 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomXmlTraceDefinition; | |
52 | import org.eclipse.tracecompass.tmf.core.project.model.TmfTraceType; | |
53 | import org.eclipse.tracecompass.tmf.core.project.model.TraceTypeHelper; | |
54 | import org.eclipse.tracecompass.tmf.core.synchronization.TimestampTransformFactory; | |
55 | import org.eclipse.tracecompass.tmf.core.timestamp.TmfTimestampFormat; | |
56 | import org.eclipse.tracecompass.tmf.core.trace.ITmfTrace; | |
57 | import org.eclipse.tracecompass.tmf.core.trace.ITmfTraceProperties; | |
2bdf0193 AM |
58 | import org.eclipse.tracecompass.tmf.core.trace.TmfTraceManager; |
59 | import org.eclipse.tracecompass.tmf.ui.editors.TmfEventsEditor; | |
60 | import org.eclipse.tracecompass.tmf.ui.properties.ReadOnlyTextPropertyDescriptor; | |
12c155f5 FC |
61 | import org.eclipse.ui.IActionFilter; |
62 | import org.eclipse.ui.views.properties.IPropertyDescriptor; | |
63 | import org.eclipse.ui.views.properties.IPropertySource2; | |
12c155f5 | 64 | |
b75a5a1b PT |
65 | import com.ibm.icu.text.DateFormat; |
66 | import com.ibm.icu.text.NumberFormat; | |
67 | ||
12c155f5 | 68 | /** |
d04ec5a7 MK |
69 | * Implementation of trace model element representing a trace. It provides |
70 | * methods to instantiate <code>ITmfTrace</code> and <code>ITmfEvent</code> as | |
71 | * well as editor ID from the trace type extension definition. | |
abbdd66a | 72 | * |
b544077e BH |
73 | * @version 1.0 |
74 | * @author Francois Chouinard | |
12c155f5 | 75 | */ |
16036bc2 | 76 | public class TmfTraceElement extends TmfCommonProjectElement implements IActionFilter, IPropertySource2 { |
12c155f5 FC |
77 | |
78 | // ------------------------------------------------------------------------ | |
79 | // Constants | |
80 | // ------------------------------------------------------------------------ | |
81 | ||
12c155f5 | 82 | // Other attributes |
b544077e BH |
83 | /** |
84 | * Bundle attribute name | |
85 | */ | |
12c155f5 | 86 | public static final String BUNDLE = "bundle"; //$NON-NLS-1$ |
b544077e BH |
87 | /** |
88 | * IsLinked attribute name. | |
89 | */ | |
12c155f5 FC |
90 | public static final String IS_LINKED = "isLinked"; //$NON-NLS-1$ |
91 | ||
92 | // Property View stuff | |
4962833a JCK |
93 | private static final String sfResourcePropertiesCategory = Messages.TmfTraceElement_ResourceProperties; |
94 | private static final String sfName = Messages.TmfTraceElement_Name; | |
95 | private static final String sfPath = Messages.TmfTraceElement_Path; | |
96 | private static final String sfLocation = Messages.TmfTraceElement_Location; | |
2918b427 | 97 | private static final String sfTraceType = Messages.TmfTraceElement_EventType; |
4962833a | 98 | private static final String sfIsLinked = Messages.TmfTraceElement_IsLinked; |
89730b51 | 99 | private static final String sfSourceLocation = Messages.TmfTraceElement_SourceLocation; |
6b44794a | 100 | private static final String sfTimeOffset = Messages.TmfTraceElement_TimeOffset; |
b75a5a1b PT |
101 | private static final String sfLastModified = Messages.TmfTraceElement_LastModified; |
102 | private static final String sfSize = Messages.TmfTraceElement_Size; | |
4962833a | 103 | private static final String sfTracePropertiesCategory = Messages.TmfTraceElement_TraceProperties; |
12c155f5 | 104 | |
253d5be1 BH |
105 | private static final ReadOnlyTextPropertyDescriptor sfNameDescriptor = new ReadOnlyTextPropertyDescriptor(sfName, sfName); |
106 | private static final ReadOnlyTextPropertyDescriptor sfPathDescriptor = new ReadOnlyTextPropertyDescriptor(sfPath, sfPath); | |
107 | private static final ReadOnlyTextPropertyDescriptor sfLocationDescriptor = new ReadOnlyTextPropertyDescriptor(sfLocation, sfLocation); | |
2918b427 | 108 | private static final ReadOnlyTextPropertyDescriptor sfTypeDescriptor = new ReadOnlyTextPropertyDescriptor(sfTraceType, sfTraceType); |
253d5be1 | 109 | private static final ReadOnlyTextPropertyDescriptor sfIsLinkedDescriptor = new ReadOnlyTextPropertyDescriptor(sfIsLinked, sfIsLinked); |
89730b51 | 110 | private static final ReadOnlyTextPropertyDescriptor sfSourceLocationDescriptor = new ReadOnlyTextPropertyDescriptor(sfSourceLocation, sfSourceLocation); |
6b44794a | 111 | private static final ReadOnlyTextPropertyDescriptor sfTimeOffsetDescriptor = new ReadOnlyTextPropertyDescriptor(sfTimeOffset, sfTimeOffset); |
b75a5a1b PT |
112 | private static final ReadOnlyTextPropertyDescriptor sfLastModifiedDescriptor = new ReadOnlyTextPropertyDescriptor(sfLastModified, sfLastModified); |
113 | private static final ReadOnlyTextPropertyDescriptor sfSizeDescriptor = new ReadOnlyTextPropertyDescriptor(sfSize, sfSize); | |
12c155f5 FC |
114 | |
115 | private static final IPropertyDescriptor[] sfDescriptors = { sfNameDescriptor, sfPathDescriptor, sfLocationDescriptor, | |
6b44794a | 116 | sfTypeDescriptor, sfIsLinkedDescriptor, sfSourceLocationDescriptor, |
b75a5a1b | 117 | sfTimeOffsetDescriptor, sfLastModifiedDescriptor, sfSizeDescriptor }; |
12c155f5 FC |
118 | |
119 | static { | |
4962833a JCK |
120 | sfNameDescriptor.setCategory(sfResourcePropertiesCategory); |
121 | sfPathDescriptor.setCategory(sfResourcePropertiesCategory); | |
122 | sfLocationDescriptor.setCategory(sfResourcePropertiesCategory); | |
123 | sfTypeDescriptor.setCategory(sfResourcePropertiesCategory); | |
124 | sfIsLinkedDescriptor.setCategory(sfResourcePropertiesCategory); | |
89730b51 | 125 | sfSourceLocationDescriptor.setCategory(sfResourcePropertiesCategory); |
6b44794a | 126 | sfTimeOffsetDescriptor.setCategory(sfResourcePropertiesCategory); |
b75a5a1b PT |
127 | sfLastModifiedDescriptor.setCategory(sfResourcePropertiesCategory); |
128 | sfSizeDescriptor.setCategory(sfResourcePropertiesCategory); | |
12c155f5 | 129 | } |
6256d8ad | 130 | |
6b44794a MK |
131 | private static final TmfTimestampFormat OFFSET_FORMAT = new TmfTimestampFormat("T.SSS SSS SSS s"); //$NON-NLS-1$ |
132 | ||
b75a5a1b PT |
133 | private static final int FOLDER_MAX_COUNT = 1024; |
134 | ||
12c155f5 FC |
135 | // ------------------------------------------------------------------------ |
136 | // Static initialization | |
137 | // ------------------------------------------------------------------------ | |
138 | ||
d04ec5a7 MK |
139 | // The mapping of available trace type IDs to their corresponding |
140 | // configuration element | |
507b1336 | 141 | private static final Map<String, IConfigurationElement> sfTraceTypeAttributes = new HashMap<>(); |
a926c25c | 142 | private static final Map<String, IConfigurationElement> sfTraceTypeUIAttributes = new HashMap<>(); |
507b1336 | 143 | private static final Map<String, IConfigurationElement> sfTraceCategories = new HashMap<>(); |
12c155f5 | 144 | |
b544077e | 145 | /** |
d04ec5a7 MK |
146 | * Initialize statically at startup by getting extensions from the platform |
147 | * extension registry. | |
b544077e | 148 | */ |
12c155f5 | 149 | public static void init() { |
a926c25c | 150 | /* Read the tmf.core "tracetype" extension point */ |
4bf17f4a | 151 | IConfigurationElement[] config = Platform.getExtensionRegistry().getConfigurationElementsFor(TmfTraceType.TMF_TRACE_TYPE_ID); |
12c155f5 | 152 | for (IConfigurationElement ce : config) { |
a926c25c AM |
153 | switch (ce.getName()) { |
154 | case TmfTraceType.TYPE_ELEM: | |
4bf17f4a | 155 | String traceTypeId = ce.getAttribute(TmfTraceType.ID_ATTR); |
12c155f5 | 156 | sfTraceTypeAttributes.put(traceTypeId, ce); |
a926c25c AM |
157 | break; |
158 | case TmfTraceType.CATEGORY_ELEM: | |
4bf17f4a | 159 | String categoryId = ce.getAttribute(TmfTraceType.ID_ATTR); |
12c155f5 | 160 | sfTraceCategories.put(categoryId, ce); |
a926c25c AM |
161 | break; |
162 | default: | |
163 | } | |
164 | } | |
165 | ||
166 | /* | |
167 | * Read the corresponding tmf.ui "tracetypeui" extension point for this | |
168 | * trace type, if it exists. | |
169 | */ | |
170 | config = Platform.getExtensionRegistry().getConfigurationElementsFor(TmfTraceTypeUIUtils.TMF_TRACE_TYPE_UI_ID); | |
171 | for (IConfigurationElement ce : config) { | |
172 | String elemName = ce.getName(); | |
173 | if (TmfTraceTypeUIUtils.TYPE_ELEM.equals(elemName)) { | |
174 | String traceType = ce.getAttribute(TmfTraceTypeUIUtils.TRACETYPE_ATTR); | |
175 | sfTraceTypeUIAttributes.put(traceType, ce); | |
12c155f5 FC |
176 | } |
177 | } | |
178 | } | |
179 | ||
b75a5a1b PT |
180 | // ------------------------------------------------------------------------ |
181 | // Classes | |
182 | // ------------------------------------------------------------------------ | |
183 | ||
184 | private class FileInfo { | |
185 | long lastModified; | |
186 | long size; | |
187 | int count; | |
188 | } | |
189 | ||
190 | // ------------------------------------------------------------------------ | |
191 | // Attributes | |
192 | // ------------------------------------------------------------------------ | |
193 | ||
194 | private FileInfo fFileInfo; | |
195 | ||
12c155f5 FC |
196 | // ------------------------------------------------------------------------ |
197 | // Constructors | |
198 | // ------------------------------------------------------------------------ | |
b544077e | 199 | /** |
d04ec5a7 MK |
200 | * Constructor. Creates trace model element under the trace folder. |
201 | * | |
202 | * @param name | |
203 | * The name of trace | |
204 | * @param trace | |
205 | * The trace resource. | |
206 | * @param parent | |
207 | * The parent element (trace folder) | |
b544077e | 208 | */ |
12c155f5 | 209 | public TmfTraceElement(String name, IResource trace, TmfTraceFolder parent) { |
339d539c | 210 | super(name, trace, parent); |
12c155f5 | 211 | } |
d04ec5a7 | 212 | |
b544077e | 213 | /** |
d04ec5a7 MK |
214 | * Constructor. Creates trace model element under the experiment folder. |
215 | * | |
216 | * @param name | |
217 | * The name of trace | |
218 | * @param trace | |
219 | * The trace resource. | |
220 | * @param parent | |
221 | * The parent element (experiment folder) | |
b544077e | 222 | */ |
12c155f5 | 223 | public TmfTraceElement(String name, IResource trace, TmfExperimentElement parent) { |
12c155f5 | 224 | super(name, trace, parent); |
f537c959 PT |
225 | } |
226 | ||
12c155f5 FC |
227 | // ------------------------------------------------------------------------ |
228 | // Operations | |
229 | // ------------------------------------------------------------------------ | |
12c155f5 | 230 | |
b544077e | 231 | /** |
d04ec5a7 MK |
232 | * Instantiate a <code>ITmfTrace</code> object based on the trace type and |
233 | * the corresponding extension. | |
abbdd66a | 234 | * |
b544077e BH |
235 | * @return the <code>ITmfTrace</code> or <code>null</code> for an error |
236 | */ | |
8f5221c2 | 237 | @Override |
6256d8ad | 238 | public ITmfTrace instantiateTrace() { |
12c155f5 | 239 | try { |
e12ecd30 BH |
240 | |
241 | // make sure that supplementary folder exists | |
242 | refreshSupplementaryFolder(); | |
243 | ||
8f5221c2 GB |
244 | if (getTraceType() != null) { |
245 | if (getTraceType().startsWith(CustomTxtTrace.class.getCanonicalName())) { | |
4bf17f4a | 246 | for (CustomTxtTraceDefinition def : CustomTxtTraceDefinition.loadAll()) { |
4b3b667b | 247 | if (getTraceType().equals(CustomTxtTrace.class.getCanonicalName() + ':' + def.categoryName+ ':' + def.definitionName)) { |
4bf17f4a | 248 | return new CustomTxtTrace(def); |
249 | } | |
250 | } | |
251 | } | |
8f5221c2 | 252 | if (getTraceType().startsWith(CustomXmlTrace.class.getCanonicalName())) { |
4bf17f4a | 253 | for (CustomXmlTraceDefinition def : CustomXmlTraceDefinition.loadAll()) { |
4b3b667b | 254 | if (getTraceType().equals(CustomXmlTrace.class.getCanonicalName() + ':' + def.categoryName+ ':' + def.definitionName)) { |
4bf17f4a | 255 | return new CustomXmlTrace(def); |
256 | } | |
257 | } | |
258 | } | |
8f5221c2 | 259 | IConfigurationElement ce = sfTraceTypeAttributes.get(getTraceType()); |
d04ec5a7 MK |
260 | if (ce == null) { |
261 | return null; | |
262 | } | |
6256d8ad | 263 | ITmfTrace trace = (ITmfTrace) ce.createExecutableExtension(TmfTraceType.TRACE_TYPE_ATTR); |
12c155f5 FC |
264 | return trace; |
265 | } | |
266 | } catch (CoreException e) { | |
8fd82db5 | 267 | Activator.getDefault().logError("Error instantiating ITmfTrace object for trace " + getName(), e); //$NON-NLS-1$ |
12c155f5 FC |
268 | } |
269 | return null; | |
270 | } | |
271 | ||
b544077e | 272 | /** |
d04ec5a7 MK |
273 | * Instantiate a <code>ITmfEvent</code> object based on the trace type and |
274 | * the corresponding extension. | |
abbdd66a | 275 | * |
b544077e BH |
276 | * @return the <code>ITmfEvent</code> or <code>null</code> for an error |
277 | */ | |
ce2388e0 | 278 | public ITmfEvent instantiateEvent() { |
12c155f5 | 279 | try { |
8f5221c2 GB |
280 | if (getTraceType() != null) { |
281 | if (getTraceType().startsWith(CustomTxtTrace.class.getCanonicalName())) { | |
4bf17f4a | 282 | for (CustomTxtTraceDefinition def : CustomTxtTraceDefinition.loadAll()) { |
4b3b667b | 283 | if (getTraceType().equals(CustomTxtTrace.class.getCanonicalName() + ':' + def.categoryName+ ':' + def.definitionName)) { |
4bf17f4a | 284 | return new CustomTxtEvent(def); |
285 | } | |
286 | } | |
287 | } | |
8f5221c2 | 288 | if (getTraceType().startsWith(CustomXmlTrace.class.getCanonicalName())) { |
4bf17f4a | 289 | for (CustomXmlTraceDefinition def : CustomXmlTraceDefinition.loadAll()) { |
4b3b667b | 290 | if (getTraceType().equals(CustomXmlTrace.class.getCanonicalName() + ':' + def.categoryName+ ':' + def.definitionName)) { |
4bf17f4a | 291 | return new CustomXmlEvent(def); |
292 | } | |
293 | } | |
294 | } | |
8f5221c2 | 295 | IConfigurationElement ce = sfTraceTypeAttributes.get(getTraceType()); |
d04ec5a7 MK |
296 | if (ce == null) { |
297 | return null; | |
298 | } | |
ce2388e0 | 299 | ITmfEvent event = (ITmfEvent) ce.createExecutableExtension(TmfTraceType.EVENT_TYPE_ATTR); |
12c155f5 FC |
300 | return event; |
301 | } | |
302 | } catch (CoreException e) { | |
8fd82db5 | 303 | Activator.getDefault().logError("Error instantiating ITmfEvent object for trace " + getName(), e); //$NON-NLS-1$ |
12c155f5 FC |
304 | } |
305 | return null; | |
306 | } | |
307 | ||
8f5221c2 | 308 | @Override |
12c155f5 | 309 | public String getEditorId() { |
8f5221c2 GB |
310 | if (getTraceType() != null) { |
311 | if (getTraceType().startsWith(CustomTxtTrace.class.getCanonicalName())) { | |
4bf17f4a | 312 | return TmfEventsEditor.ID; |
313 | } | |
8f5221c2 | 314 | if (getTraceType().startsWith(CustomXmlTrace.class.getCanonicalName())) { |
4bf17f4a | 315 | return TmfEventsEditor.ID; |
316 | } | |
8f5221c2 | 317 | IConfigurationElement ce = sfTraceTypeUIAttributes.get(getTraceType()); |
a926c25c AM |
318 | if (ce == null) { |
319 | /* This trace type does not define UI attributes */ | |
320 | return null; | |
321 | } | |
322 | IConfigurationElement[] defaultEditorCE = ce.getChildren(TmfTraceTypeUIUtils.DEFAULT_EDITOR_ELEM); | |
12c155f5 | 323 | if (defaultEditorCE.length == 1) { |
4bf17f4a | 324 | return defaultEditorCE[0].getAttribute(TmfTraceType.ID_ATTR); |
12c155f5 FC |
325 | } |
326 | } | |
327 | return null; | |
328 | } | |
e12ecd30 | 329 | |
c4c81d91 | 330 | /** |
d04ec5a7 MK |
331 | * Returns the file resource used to store bookmarks after creating it if |
332 | * necessary. If the trace resource is a file, it is returned directly. If | |
333 | * the trace resource is a folder, a linked file is returned. The file will | |
334 | * be created if it does not exist. | |
335 | * | |
c4c81d91 | 336 | * @return the bookmarks file |
d04ec5a7 MK |
337 | * @throws CoreException |
338 | * if the bookmarks file cannot be created | |
c4c81d91 PT |
339 | * @since 2.0 |
340 | */ | |
8f5221c2 | 341 | @Override |
81fe3479 PT |
342 | public IFile createBookmarksFile() throws CoreException { |
343 | IFile file = getBookmarksFile(); | |
344 | if (fResource instanceof IFolder) { | |
58ffe079 | 345 | return createBookmarksFile(getProject().getTracesFolder().getResource(), ITmfEventsEditorConstants.TRACE_EDITOR_INPUT_TYPE); |
81fe3479 PT |
346 | } |
347 | return file; | |
348 | } | |
349 | ||
350 | /** | |
d04ec5a7 MK |
351 | * Returns the file resource used to store bookmarks. The file may not |
352 | * exist. | |
353 | * | |
81fe3479 PT |
354 | * @return the bookmarks file |
355 | * @since 2.0 | |
356 | */ | |
8f5221c2 | 357 | @Override |
81fe3479 | 358 | public IFile getBookmarksFile() { |
c4c81d91 PT |
359 | IFile file = null; |
360 | if (fResource instanceof IFile) { | |
361 | file = (IFile) fResource; | |
362 | } else if (fResource instanceof IFolder) { | |
c4c81d91 PT |
363 | final IFolder folder = (IFolder) fResource; |
364 | file = folder.getFile(getName() + '_'); | |
c4c81d91 PT |
365 | } |
366 | return file; | |
367 | } | |
368 | ||
e12ecd30 | 369 | /** |
d04ec5a7 MK |
370 | * Returns the <code>TmfTraceElement</code> located under the |
371 | * <code>TmfTracesFolder</code>. | |
6256d8ad | 372 | * |
d04ec5a7 MK |
373 | * @return <code>this</code> if this element is under the |
374 | * <code>TmfTracesFolder</code> else the corresponding | |
375 | * <code>TmfTraceElement</code> if this element is under | |
e12ecd30 BH |
376 | * <code>TmfExperimentElement</code>. |
377 | */ | |
5e4bf87d | 378 | public TmfTraceElement getElementUnderTraceFolder() { |
e12ecd30 | 379 | |
d04ec5a7 MK |
380 | // If trace is under an experiment, return original trace from the |
381 | // traces folder | |
5e4bf87d BH |
382 | if (getParent() instanceof TmfExperimentElement) { |
383 | for (TmfTraceElement aTrace : getProject().getTracesFolder().getTraces()) { | |
339d539c | 384 | if (aTrace.getElementPath().equals(getElementPath())) { |
5e4bf87d BH |
385 | return aTrace; |
386 | } | |
387 | } | |
388 | } | |
389 | return this; | |
390 | } | |
6256d8ad | 391 | |
8f5221c2 GB |
392 | @Override |
393 | public String getTypeName() { | |
394 | return Messages.TmfTraceElement_TypeName; | |
395 | } | |
396 | ||
12c155f5 FC |
397 | // ------------------------------------------------------------------------ |
398 | // IActionFilter | |
399 | // ------------------------------------------------------------------------ | |
400 | ||
401 | @Override | |
402 | public boolean testAttribute(Object target, String name, String value) { | |
403 | if (name.equals(IS_LINKED)) { | |
404 | boolean isLinked = getResource().isLinked(); | |
405 | return Boolean.toString(isLinked).equals(value); | |
406 | } | |
407 | return false; | |
408 | } | |
409 | ||
12c155f5 FC |
410 | // ------------------------------------------------------------------------ |
411 | // IPropertySource2 | |
412 | // ------------------------------------------------------------------------ | |
413 | ||
414 | @Override | |
415 | public Object getEditableValue() { | |
416 | return null; | |
417 | } | |
418 | ||
4962833a JCK |
419 | /** |
420 | * Get the trace properties of this traceElement if the corresponding trace | |
421 | * is opened in an editor | |
422 | * | |
423 | * @return a map with the names and values of the trace properties | |
424 | * respectively as keys and values | |
425 | */ | |
426 | private Map<String, String> getTraceProperties() { | |
8d534d69 JCK |
427 | for (ITmfTrace openedTrace : TmfTraceManager.getInstance().getOpenedTraces()) { |
428 | for (ITmfTrace singleTrace : TmfTraceManager.getTraceSet(openedTrace)) { | |
6e74bac3 | 429 | if (getElementUnderTraceFolder().getResource().equals(singleTrace.getResource())) { |
8d534d69 JCK |
430 | if (singleTrace instanceof ITmfTraceProperties) { |
431 | ITmfTraceProperties traceProperties = (ITmfTraceProperties) singleTrace; | |
432 | return traceProperties.getTraceProperties(); | |
433 | } | |
4962833a JCK |
434 | } |
435 | } | |
436 | } | |
507b1336 | 437 | return new HashMap<>(); |
4962833a JCK |
438 | } |
439 | ||
12c155f5 FC |
440 | @Override |
441 | public IPropertyDescriptor[] getPropertyDescriptors() { | |
4962833a JCK |
442 | Map<String, String> traceProperties = getTraceProperties(); |
443 | if (!traceProperties.isEmpty()) { | |
444 | IPropertyDescriptor[] propertyDescriptorArray = new IPropertyDescriptor[traceProperties.size() + sfDescriptors.length]; | |
445 | int index = 0; | |
446 | for (Map.Entry<String, String> varName : traceProperties.entrySet()) { | |
447 | ReadOnlyTextPropertyDescriptor descriptor = new ReadOnlyTextPropertyDescriptor(this.getName() + "_" + varName.getKey(), varName.getKey()); //$NON-NLS-1$ | |
448 | descriptor.setCategory(sfTracePropertiesCategory); | |
449 | propertyDescriptorArray[index] = descriptor; | |
450 | index++; | |
451 | } | |
452 | for (int i = 0; i < sfDescriptors.length; i++) { | |
453 | propertyDescriptorArray[index] = sfDescriptors[i]; | |
454 | index++; | |
455 | } | |
456 | return propertyDescriptorArray; | |
457 | } | |
77fdc5df | 458 | return Arrays.copyOf(sfDescriptors, sfDescriptors.length); |
12c155f5 FC |
459 | } |
460 | ||
461 | @Override | |
462 | public Object getPropertyValue(Object id) { | |
463 | ||
ce2388e0 | 464 | if (sfName.equals(id)) { |
12c155f5 | 465 | return getName(); |
ce2388e0 | 466 | } |
12c155f5 | 467 | |
ce2388e0 | 468 | if (sfPath.equals(id)) { |
12c155f5 | 469 | return getPath().toString(); |
ce2388e0 | 470 | } |
12c155f5 | 471 | |
ce2388e0 | 472 | if (sfLocation.equals(id)) { |
13f2a21a | 473 | return URIUtil.toUnencodedString(getLocation()); |
ce2388e0 | 474 | } |
12c155f5 | 475 | |
933ff94f PT |
476 | if (sfIsLinked.equals(id)) { |
477 | return Boolean.valueOf(getResource().isLinked()).toString(); | |
478 | } | |
479 | ||
89730b51 PT |
480 | if (sfSourceLocation.equals(id)) { |
481 | try { | |
482 | String sourceLocation = getElementUnderTraceFolder().getResource().getPersistentProperty(TmfCommonConstants.SOURCE_LOCATION); | |
483 | if (sourceLocation != null) { | |
484 | return sourceLocation; | |
485 | } | |
486 | } catch (CoreException e) { | |
487 | } | |
488 | return ""; //$NON-NLS-1$ | |
ce2388e0 | 489 | } |
12c155f5 | 490 | |
b75a5a1b PT |
491 | if (sfLastModified.equals(id)) { |
492 | FileInfo fileInfo = getFileInfo(); | |
493 | if (fileInfo == null) { | |
494 | return ""; //$NON-NLS-1$ | |
495 | } | |
496 | long date = fileInfo.lastModified; | |
497 | DateFormat format = DateFormat.getDateTimeInstance(DateFormat.LONG, DateFormat.MEDIUM); | |
498 | return format.format(new Date(date)); | |
499 | } | |
500 | ||
501 | if (sfSize.equals(id)) { | |
502 | FileInfo fileInfo = getFileInfo(); | |
503 | if (fileInfo == null) { | |
504 | return ""; //$NON-NLS-1$ | |
505 | } | |
506 | if (getResource() instanceof IFolder) { | |
507 | if (fileInfo.count <= FOLDER_MAX_COUNT) { | |
508 | return NLS.bind(Messages.TmfTraceElement_FolderSizeString, | |
509 | NumberFormat.getInstance().format(fileInfo.size), fileInfo.count); | |
510 | } | |
511 | return NLS.bind(Messages.TmfTraceElement_FolderSizeOverflowString, | |
512 | NumberFormat.getInstance().format(fileInfo.size), FOLDER_MAX_COUNT); | |
513 | } | |
514 | return NLS.bind(Messages.TmfTraceElement_FileSizeString, NumberFormat.getInstance().format(fileInfo.size)); | |
515 | } | |
516 | ||
2918b427 | 517 | if (sfTraceType.equals(id)) { |
8f5221c2 | 518 | if (getTraceType() != null) { |
4b3b667b | 519 | TraceTypeHelper helper = TmfTraceType.getTraceType(getTraceType()); |
2918b427 BH |
520 | if (helper != null) { |
521 | return helper.getCategoryName() + " : " + helper.getName(); //$NON-NLS-1$ | |
522 | } | |
12c155f5 | 523 | } |
339d539c | 524 | return ""; //$NON-NLS-1$ |
12c155f5 FC |
525 | } |
526 | ||
6b44794a MK |
527 | if (sfTimeOffset.equals(id)) { |
528 | long offset = TimestampTransformFactory.getTimestampTransform(getElementUnderTraceFolder().getResource()).transform(0); | |
529 | if (offset != 0) { | |
530 | return OFFSET_FORMAT.format(offset); | |
531 | } | |
532 | return ""; //$NON-NLS-1$ | |
533 | } | |
534 | ||
4962833a JCK |
535 | Map<String, String> traceProperties = getTraceProperties(); |
536 | if (id != null && !traceProperties.isEmpty()) { | |
537 | String key = (String) id; | |
e48bd720 | 538 | key = key.substring(this.getName().length() + 1); // remove name_ |
4962833a JCK |
539 | String value = traceProperties.get(key); |
540 | return value; | |
541 | } | |
542 | ||
12c155f5 FC |
543 | return null; |
544 | } | |
545 | ||
b75a5a1b PT |
546 | private FileInfo getFileInfo() { |
547 | /* FileInfo is needed for both 'last modified' and 'size' properties. | |
548 | * It is freshly computed for one, and reused for the other, then | |
549 | * cleared so that the information can be refreshed the next time. | |
550 | */ | |
551 | FileInfo fileInfo; | |
552 | if (fFileInfo == null) { | |
553 | try { | |
554 | fileInfo = computeFileInfo(new FileInfo(), getResource()); | |
555 | } catch (CoreException e) { | |
556 | return null; | |
557 | } | |
558 | fFileInfo = fileInfo; | |
559 | } else { | |
560 | fileInfo = fFileInfo; | |
561 | fFileInfo = null; | |
562 | } | |
563 | return fileInfo; | |
564 | } | |
565 | ||
566 | private FileInfo computeFileInfo(FileInfo fileInfo, IResource resource) throws CoreException { | |
567 | if (fileInfo == null || fileInfo.count > FOLDER_MAX_COUNT) { | |
568 | return fileInfo; | |
569 | } | |
570 | if (resource instanceof IFolder) { | |
571 | IFolder folder = (IFolder) resource; | |
572 | for (IResource member : folder.members()) { | |
573 | computeFileInfo(fileInfo, member); | |
574 | } | |
575 | return fileInfo; | |
576 | } | |
577 | IFileInfo info = EFS.getStore(resource.getLocationURI()).fetchInfo(); | |
578 | fileInfo.lastModified = Math.max(fileInfo.lastModified, info.getLastModified()); | |
579 | fileInfo.size += info.getLength(); | |
580 | fileInfo.count++; | |
581 | return fileInfo; | |
582 | } | |
583 | ||
12c155f5 FC |
584 | @Override |
585 | public void resetPropertyValue(Object id) { | |
586 | } | |
587 | ||
588 | @Override | |
589 | public void setPropertyValue(Object id, Object value) { | |
590 | } | |
591 | ||
592 | @Override | |
593 | public boolean isPropertyResettable(Object id) { | |
594 | return false; | |
595 | } | |
596 | ||
597 | @Override | |
598 | public boolean isPropertySet(Object id) { | |
599 | return false; | |
600 | } | |
601 | ||
beb19106 GB |
602 | /** |
603 | * Copy this trace in the trace folder. No other parameters are mentioned so | |
604 | * the trace is copied in this element's project trace folder | |
605 | * | |
f3cbd37d | 606 | * @param newName |
beb19106 GB |
607 | * The new trace name |
608 | * @return the new Resource object | |
609 | * @since 2.0 | |
610 | */ | |
f3cbd37d PT |
611 | public TmfTraceElement copy(String newName) { |
612 | TmfTraceFolder folder = (TmfTraceFolder) getParent(); | |
613 | IResource res = super.copy(newName, false); | |
614 | for (TmfTraceElement trace : folder.getTraces()) { | |
615 | if (trace.getResource().equals(res)) { | |
616 | return trace; | |
617 | } | |
618 | } | |
619 | return null; | |
beb19106 GB |
620 | } |
621 | ||
a72a6830 PT |
622 | /** |
623 | * Close opened editors associated with this trace. | |
4962833a | 624 | * |
a72a6830 PT |
625 | * @since 2.0 |
626 | */ | |
8f5221c2 | 627 | @Override |
a72a6830 | 628 | public void closeEditors() { |
8f5221c2 | 629 | super.closeEditors(); |
a72a6830 PT |
630 | |
631 | // Close experiments that contain the trace if open | |
632 | if (getParent() instanceof TmfTraceFolder) { | |
339d539c PT |
633 | TmfExperimentFolder experimentsFolder = getProject().getExperimentsFolder(); |
634 | for (TmfExperimentElement experiment : experimentsFolder.getExperiments()) { | |
635 | for (TmfTraceElement trace : experiment.getTraces()) { | |
636 | if (trace.getElementPath().equals(getElementPath())) { | |
637 | experiment.closeEditors(); | |
a72a6830 PT |
638 | break; |
639 | } | |
640 | } | |
641 | } | |
642 | } else if (getParent() instanceof TmfExperimentElement) { | |
643 | TmfExperimentElement experiment = (TmfExperimentElement) getParent(); | |
644 | experiment.closeEditors(); | |
645 | } | |
646 | } | |
8dcdf263 | 647 | |
6e651d8b MAL |
648 | /** |
649 | * Delete the trace resource, remove it from experiments and delete its | |
650 | * supplementary files | |
651 | * | |
652 | * @param progressMonitor | |
653 | * a progress monitor, or null if progress reporting is not | |
654 | * desired | |
655 | * | |
656 | * @throws CoreException | |
657 | * thrown when IResource.delete fails | |
658 | * @since 2.2 | |
659 | */ | |
660 | public void delete(IProgressMonitor progressMonitor) throws CoreException { | |
30c7c7d5 MAL |
661 | // Close editors in UI Thread |
662 | Display.getDefault().syncExec(new Runnable() { | |
663 | @Override | |
664 | public void run() { | |
665 | closeEditors(); | |
666 | } | |
667 | }); | |
6e651d8b MAL |
668 | |
669 | IPath path = fResource.getLocation(); | |
2bb6a6b8 BH |
670 | if (path != null) { |
671 | if (getParent() instanceof TmfTraceFolder) { | |
672 | TmfExperimentFolder experimentFolder = getProject().getExperimentsFolder(); | |
673 | ||
674 | // Propagate the removal to traces | |
339d539c PT |
675 | for (TmfExperimentElement experiment : experimentFolder.getExperiments()) { |
676 | List<TmfTraceElement> toRemove = new LinkedList<>(); | |
677 | for (TmfTraceElement trace : experiment.getTraces()) { | |
678 | if (trace.getElementPath().equals(getElementPath())) { | |
679 | toRemove.add(trace); | |
2bb6a6b8 BH |
680 | } |
681 | } | |
339d539c PT |
682 | for (TmfTraceElement child : toRemove) { |
683 | experiment.removeTrace(child); | |
6e651d8b MAL |
684 | } |
685 | } | |
6e651d8b | 686 | |
2bb6a6b8 BH |
687 | // Delete supplementary files |
688 | deleteSupplementaryFolder(); | |
689 | ||
690 | } else if (getParent() instanceof TmfExperimentElement) { | |
691 | TmfExperimentElement experimentElement = (TmfExperimentElement) getParent(); | |
692 | experimentElement.removeTrace(this); | |
693 | } | |
6e651d8b MAL |
694 | } |
695 | ||
696 | // Finally, delete the trace | |
697 | fResource.delete(true, progressMonitor); | |
698 | } | |
699 | ||
12c155f5 | 700 | } |