Commit 363c533e072fa7adf5be7e4947aface2410e1c11
1 parent
69773da3
Exists in
master
Testes de módulo de gerenciamento.
Showing
8 changed files
with
307 additions
and
37 deletions
Show diff stats
impl/core/src/main/java/br/gov/frameworkdemoiselle/management/internal/MonitoringManager.java
@@ -63,7 +63,10 @@ public class MonitoringManager { | @@ -63,7 +63,10 @@ public class MonitoringManager { | ||
63 | } | 63 | } |
64 | 64 | ||
65 | /** | 65 | /** |
66 | - * Invoke an operation over a managed type. | 66 | + * <p>Invoke an operation over a managed type.</p> |
67 | + * | ||
68 | + * <p>This method is not thread-safe, it's the user's responsibility to | ||
69 | + * make the operations of the managed type synchronized if necessary.</p> | ||
67 | * | 70 | * |
68 | * @param managedType | 71 | * @param managedType |
69 | * A type annotated with {@link Managed}. This method will create | 72 | * A type annotated with {@link Managed}. This method will create |
@@ -81,7 +84,7 @@ public class MonitoringManager { | @@ -81,7 +84,7 @@ public class MonitoringManager { | ||
81 | * In case the operation doesn't exist or have a different | 84 | * In case the operation doesn't exist or have a different |
82 | * signature | 85 | * signature |
83 | */ | 86 | */ |
84 | - public synchronized Object invoke(ManagedType managedType, String actionName, | 87 | + public Object invoke(ManagedType managedType, String actionName, |
85 | Object[] params) { | 88 | Object[] params) { |
86 | if ( managedTypes.contains(managedType) ) { | 89 | if ( managedTypes.contains(managedType) ) { |
87 | activateContexts(managedType.getType()); | 90 | activateContexts(managedType.getType()); |
@@ -112,14 +115,17 @@ public class MonitoringManager { | @@ -112,14 +115,17 @@ public class MonitoringManager { | ||
112 | } | 115 | } |
113 | 116 | ||
114 | /** | 117 | /** |
115 | - * Retrieve the current value of a property from a managed type. Properties | ||
116 | - * are attributes annotated with {@link Property}. | 118 | + * <p>Retrieve the current value of a property from a managed type. Properties |
119 | + * are attributes annotated with {@link Property}.</p> | ||
120 | + * | ||
121 | + * <p>This method is not thread-safe, it's the user's responsibility to | ||
122 | + * create the property's access methods from the managed type synchronized if necessary.</p> | ||
117 | * | 123 | * |
118 | * @param managedType The type that has the property the client wants to know the value of. | 124 | * @param managedType The type that has the property the client wants to know the value of. |
119 | * @param propertyName The name of the property | 125 | * @param propertyName The name of the property |
120 | * @return The current value of the property | 126 | * @return The current value of the property |
121 | */ | 127 | */ |
122 | - public synchronized Object getProperty(ManagedType managedType, String propertyName) { | 128 | + public Object getProperty(ManagedType managedType, String propertyName) { |
123 | 129 | ||
124 | if ( managedTypes.contains(managedType) ) { | 130 | if ( managedTypes.contains(managedType) ) { |
125 | Method getterMethod = managedType.getFields().get(propertyName).getGetterMethod(); | 131 | Method getterMethod = managedType.getFields().get(propertyName).getGetterMethod(); |
@@ -153,14 +159,17 @@ public class MonitoringManager { | @@ -153,14 +159,17 @@ public class MonitoringManager { | ||
153 | } | 159 | } |
154 | 160 | ||
155 | /** | 161 | /** |
156 | - * Sets a new value for a property contained inside a managed type. A property | ||
157 | - * is an attribute annotated with {@link Property}. | 162 | + * <p>Sets a new value for a property contained inside a managed type. A property |
163 | + * is an attribute annotated with {@link Property}.</p> | ||
164 | + * | ||
165 | + * <p>This method is not thread-safe, it's the user's responsibility to | ||
166 | + * create the property's access methods from the managed type synchronized if necessary.</p> | ||
158 | * | 167 | * |
159 | * @param managedType The type that has access to the property | 168 | * @param managedType The type that has access to the property |
160 | * @param propertyName The name of the property | 169 | * @param propertyName The name of the property |
161 | * @param newValue The new value of the property | 170 | * @param newValue The new value of the property |
162 | */ | 171 | */ |
163 | - public synchronized void setProperty(ManagedType managedType, String propertyName, | 172 | + public void setProperty(ManagedType managedType, String propertyName, |
164 | Object newValue) { | 173 | Object newValue) { |
165 | 174 | ||
166 | if ( managedTypes.contains(managedType) ) { | 175 | if ( managedTypes.contains(managedType) ) { |
impl/core/src/test/java/management/ManagedClassStore.java
@@ -1,25 +0,0 @@ | @@ -1,25 +0,0 @@ | ||
1 | -package management; | ||
2 | - | ||
3 | -import java.util.ArrayList; | ||
4 | -import java.util.Collection; | ||
5 | -import java.util.List; | ||
6 | - | ||
7 | -import javax.enterprise.context.ApplicationScoped; | ||
8 | - | ||
9 | -import br.gov.frameworkdemoiselle.management.internal.ManagedType; | ||
10 | - | ||
11 | -@ApplicationScoped | ||
12 | -public class ManagedClassStore { | ||
13 | - | ||
14 | - private List<ManagedType> managedTypes = new ArrayList<ManagedType>(); | ||
15 | - | ||
16 | - | ||
17 | - public List<ManagedType> getManagedTypes() { | ||
18 | - return managedTypes; | ||
19 | - } | ||
20 | - | ||
21 | - public void addManagedTypes(Collection<ManagedType> managedTypes){ | ||
22 | - this.managedTypes.addAll(managedTypes); | ||
23 | - } | ||
24 | - | ||
25 | -} |
impl/core/src/test/java/management/ManagementBootstrapTestCase.java
@@ -6,6 +6,7 @@ import java.util.List; | @@ -6,6 +6,7 @@ import java.util.List; | ||
6 | import management.testclasses.DummyManagedClass; | 6 | import management.testclasses.DummyManagedClass; |
7 | import management.testclasses.DummyManagedClassPropertyError; | 7 | import management.testclasses.DummyManagedClassPropertyError; |
8 | import management.testclasses.DummyManagementExtension; | 8 | import management.testclasses.DummyManagementExtension; |
9 | +import management.testclasses.ManagedClassStore; | ||
9 | 10 | ||
10 | import org.jboss.arquillian.container.test.api.Deployer; | 11 | import org.jboss.arquillian.container.test.api.Deployer; |
11 | import org.jboss.arquillian.container.test.api.Deployment; | 12 | import org.jboss.arquillian.container.test.api.Deployment; |
@@ -44,7 +45,7 @@ public class ManagementBootstrapTestCase { | @@ -44,7 +45,7 @@ public class ManagementBootstrapTestCase { | ||
44 | new File("src/main/resources/META-INF/services/javax.enterprise.inject.spi.Extension"), | 45 | new File("src/main/resources/META-INF/services/javax.enterprise.inject.spi.Extension"), |
45 | "services/javax.enterprise.inject.spi.Extension") | 46 | "services/javax.enterprise.inject.spi.Extension") |
46 | .addPackages(false, ManagementBootstrapTestCase.class.getPackage()) | 47 | .addPackages(false, ManagementBootstrapTestCase.class.getPackage()) |
47 | - .addClasses(DummyManagementExtension.class,DummyManagedClass.class); | 48 | + .addClasses(DummyManagementExtension.class,DummyManagedClass.class,ManagedClassStore.class); |
48 | } | 49 | } |
49 | 50 | ||
50 | /** | 51 | /** |
@@ -63,7 +64,7 @@ public class ManagementBootstrapTestCase { | @@ -63,7 +64,7 @@ public class ManagementBootstrapTestCase { | ||
63 | new File("src/main/resources/META-INF/services/javax.enterprise.inject.spi.Extension"), | 64 | new File("src/main/resources/META-INF/services/javax.enterprise.inject.spi.Extension"), |
64 | "services/javax.enterprise.inject.spi.Extension") | 65 | "services/javax.enterprise.inject.spi.Extension") |
65 | .addPackages(false, ManagementBootstrapTestCase.class.getPackage()) | 66 | .addPackages(false, ManagementBootstrapTestCase.class.getPackage()) |
66 | - .addClasses(DummyManagementExtension.class,DummyManagedClassPropertyError.class); | 67 | + .addClasses(DummyManagementExtension.class,DummyManagedClassPropertyError.class,ManagedClassStore.class); |
67 | } | 68 | } |
68 | 69 | ||
69 | /** | 70 | /** |
impl/core/src/test/java/management/ManagementTestCase.java
0 → 100644
@@ -0,0 +1,128 @@ | @@ -0,0 +1,128 @@ | ||
1 | +package management; | ||
2 | + | ||
3 | +import java.io.File; | ||
4 | + | ||
5 | +import junit.framework.Assert; | ||
6 | +import management.testclasses.DummyManagedClass; | ||
7 | +import management.testclasses.DummyManagementExtension; | ||
8 | +import management.testclasses.ManagedClassStore; | ||
9 | + | ||
10 | +import org.jboss.arquillian.container.test.api.Deployment; | ||
11 | +import org.jboss.arquillian.junit.Arquillian; | ||
12 | +import org.jboss.shrinkwrap.api.ShrinkWrap; | ||
13 | +import org.jboss.shrinkwrap.api.asset.FileAsset; | ||
14 | +import org.jboss.shrinkwrap.api.spec.JavaArchive; | ||
15 | +import org.junit.Test; | ||
16 | +import org.junit.runner.RunWith; | ||
17 | + | ||
18 | +import test.LocaleProducer; | ||
19 | +import br.gov.frameworkdemoiselle.DemoiselleException; | ||
20 | +import br.gov.frameworkdemoiselle.util.Beans; | ||
21 | + | ||
22 | +/** | ||
23 | + * Test case that simulates a management extension and tests if properties and operations on a managed class can be | ||
24 | + * easily accessed and invoked. | ||
25 | + * | ||
26 | + * @author serpro | ||
27 | + */ | ||
28 | +@RunWith(Arquillian.class) | ||
29 | +public class ManagementTestCase { | ||
30 | + | ||
31 | + @Deployment | ||
32 | + public static JavaArchive createMultithreadedDeployment() { | ||
33 | + return ShrinkWrap | ||
34 | + .create(JavaArchive.class) | ||
35 | + .addClass(LocaleProducer.class) | ||
36 | + .addPackages(true, "br") | ||
37 | + .addAsResource(new FileAsset(new File("src/test/resources/test/beans.xml")), "beans.xml") | ||
38 | + .addAsManifestResource( | ||
39 | + new File("src/main/resources/META-INF/services/javax.enterprise.inject.spi.Extension"), | ||
40 | + "services/javax.enterprise.inject.spi.Extension") | ||
41 | + .addPackages(false, ManagementTestCase.class.getPackage()) | ||
42 | + .addClasses(DummyManagementExtension.class, DummyManagedClass.class, ManagedClassStore.class); | ||
43 | + } | ||
44 | + | ||
45 | + @Test | ||
46 | + public void testReadProperty() { | ||
47 | + DummyManagedClass managedClass = Beans.getReference(DummyManagedClass.class); | ||
48 | + managedClass.setName("Test Name"); | ||
49 | + | ||
50 | + // store é nossa extensão de gerenciamento falsa, então estamos testando um "cliente" acessando | ||
51 | + // nosso tipo gerenciado DummyManagedClass remotamente. | ||
52 | + ManagedClassStore store = Beans.getReference(ManagedClassStore.class); | ||
53 | + Object name = store.getProperty(DummyManagedClass.class, "name"); | ||
54 | + Assert.assertEquals("Test Name", name); | ||
55 | + } | ||
56 | + | ||
57 | + @Test | ||
58 | + public void testWriteProperty() { | ||
59 | + // store é nossa extensão de gerenciamento falsa, então estamos testando um "cliente" definindo | ||
60 | + // um novo valor em uma propriedade de nosso tipo gerenciado DummyManagedClass remotamente. | ||
61 | + ManagedClassStore store = Beans.getReference(ManagedClassStore.class); | ||
62 | + store.setProperty(DummyManagedClass.class, "name", "Test Name"); | ||
63 | + | ||
64 | + DummyManagedClass managedClass = Beans.getReference(DummyManagedClass.class); | ||
65 | + Assert.assertEquals("Test Name", managedClass.getName()); | ||
66 | + } | ||
67 | + | ||
68 | + @Test | ||
69 | + public void testReadAWriteOnly() { | ||
70 | + | ||
71 | + ManagedClassStore store = Beans.getReference(ManagedClassStore.class); | ||
72 | + | ||
73 | + try { | ||
74 | + store.getProperty(DummyManagedClass.class, "writeOnlyProperty"); | ||
75 | + Assert.fail(); | ||
76 | + } catch (DemoiselleException de) { | ||
77 | + // SUCCESS | ||
78 | + } | ||
79 | + | ||
80 | + } | ||
81 | + | ||
82 | + @Test | ||
83 | + public void testWriteAReadOnly() { | ||
84 | + | ||
85 | + ManagedClassStore store = Beans.getReference(ManagedClassStore.class); | ||
86 | + | ||
87 | + try { | ||
88 | + store.setProperty(DummyManagedClass.class, "readOnlyProperty", "New Value"); | ||
89 | + Assert.fail(); | ||
90 | + } catch (DemoiselleException de) { | ||
91 | + // SUCCESS | ||
92 | + } | ||
93 | + | ||
94 | + } | ||
95 | + | ||
96 | + @Test | ||
97 | + public void testInvokeOperation() { | ||
98 | + | ||
99 | + ManagedClassStore store = Beans.getReference(ManagedClassStore.class); | ||
100 | + | ||
101 | + try { | ||
102 | + store.setProperty(DummyManagedClass.class, "firstFactor", new Integer(10)); | ||
103 | + store.setProperty(DummyManagedClass.class, "secondFactor", new Integer(15)); | ||
104 | + Integer response = (Integer) store.invoke(DummyManagedClass.class, "sumFactors"); | ||
105 | + Assert.assertEquals(new Integer(25), response); | ||
106 | + } catch (DemoiselleException de) { | ||
107 | + Assert.fail(de.getMessage()); | ||
108 | + } | ||
109 | + | ||
110 | + } | ||
111 | + | ||
112 | + @Test | ||
113 | + public void testInvokeNonAnnotatedOperation() { | ||
114 | + | ||
115 | + ManagedClassStore store = Beans.getReference(ManagedClassStore.class); | ||
116 | + | ||
117 | + try { | ||
118 | + // O método "nonOperationAnnotatedMethod" existe na classe DummyManagedClass, mas não está anotado como | ||
119 | + // "@Operation", então | ||
120 | + // ela não pode ser exposta para extensões. | ||
121 | + store.invoke(DummyManagedClass.class, "nonOperationAnnotatedMethod"); | ||
122 | + Assert.fail(); | ||
123 | + } catch (DemoiselleException de) { | ||
124 | + // SUCCESS | ||
125 | + } | ||
126 | + | ||
127 | + } | ||
128 | +} |
impl/core/src/test/java/management/NotificationTestCase.java
@@ -53,7 +53,7 @@ public class NotificationTestCase { | @@ -53,7 +53,7 @@ public class NotificationTestCase { | ||
53 | .addAsManifestResource( | 53 | .addAsManifestResource( |
54 | new File("src/main/resources/META-INF/services/javax.enterprise.inject.spi.Extension"), | 54 | new File("src/main/resources/META-INF/services/javax.enterprise.inject.spi.Extension"), |
55 | "services/javax.enterprise.inject.spi.Extension") | 55 | "services/javax.enterprise.inject.spi.Extension") |
56 | - .addPackages(false, ManagementBootstrapTestCase.class.getPackage()) | 56 | + .addPackages(false, NotificationTestCase.class.getPackage()) |
57 | .addClasses(DummyNotificationListener.class,DummyManagedClass.class); | 57 | .addClasses(DummyNotificationListener.class,DummyManagedClass.class); |
58 | } | 58 | } |
59 | 59 |
impl/core/src/test/java/management/testclasses/DummyManagedClass.java
@@ -12,15 +12,24 @@ import br.gov.frameworkdemoiselle.management.annotation.validation.AllowedValues | @@ -12,15 +12,24 @@ import br.gov.frameworkdemoiselle.management.annotation.validation.AllowedValues | ||
12 | public class DummyManagedClass { | 12 | public class DummyManagedClass { |
13 | 13 | ||
14 | @Property | 14 | @Property |
15 | + private String name; | ||
16 | + | ||
17 | + @Property | ||
15 | @AllowedValues(allows={"f","m","F","M"},valueType=ValueType.INTEGER) | 18 | @AllowedValues(allows={"f","m","F","M"},valueType=ValueType.INTEGER) |
16 | private Integer id; | 19 | private Integer id; |
17 | 20 | ||
18 | @Property | 21 | @Property |
22 | + private Integer firstFactor , secondFactor; | ||
23 | + | ||
24 | + @Property | ||
19 | private String uuid; | 25 | private String uuid; |
20 | 26 | ||
21 | @Property | 27 | @Property |
22 | private String writeOnlyProperty; | 28 | private String writeOnlyProperty; |
23 | 29 | ||
30 | + @Property | ||
31 | + private String readOnlyProperty = "Default Value"; | ||
32 | + | ||
24 | /** | 33 | /** |
25 | * Propriedade para testar detecção de métodos GET e SET quando propriedade tem apenas uma letra. | 34 | * Propriedade para testar detecção de métodos GET e SET quando propriedade tem apenas uma letra. |
26 | */ | 35 | */ |
@@ -71,4 +80,87 @@ public class DummyManagedClass { | @@ -71,4 +80,87 @@ public class DummyManagedClass { | ||
71 | this.uuid = UUID.randomUUID().toString(); | 80 | this.uuid = UUID.randomUUID().toString(); |
72 | return this.uuid; | 81 | return this.uuid; |
73 | } | 82 | } |
83 | + | ||
84 | + | ||
85 | + public String getName() { | ||
86 | + return name; | ||
87 | + } | ||
88 | + | ||
89 | + | ||
90 | + public void setName(String name) { | ||
91 | + this.name = name; | ||
92 | + } | ||
93 | + | ||
94 | + | ||
95 | + public String getReadOnlyProperty() { | ||
96 | + return readOnlyProperty; | ||
97 | + } | ||
98 | + | ||
99 | + | ||
100 | + public Integer getFirstFactor() { | ||
101 | + return firstFactor; | ||
102 | + } | ||
103 | + | ||
104 | + | ||
105 | + public void setFirstFactor(Integer firstFactor) { | ||
106 | + this.firstFactor = firstFactor; | ||
107 | + } | ||
108 | + | ||
109 | + | ||
110 | + public Integer getSecondFactor() { | ||
111 | + return secondFactor; | ||
112 | + } | ||
113 | + | ||
114 | + | ||
115 | + public void setSecondFactor(Integer secondFactor) { | ||
116 | + this.secondFactor = secondFactor; | ||
117 | + } | ||
118 | + | ||
119 | + @Operation | ||
120 | + public Integer calculateFactorsNonSynchronized(Integer firstFactor , Integer secondFactor){ | ||
121 | + setFirstFactor(firstFactor); | ||
122 | + setSecondFactor(secondFactor); | ||
123 | + | ||
124 | + try { | ||
125 | + int temp = firstFactor + secondFactor; | ||
126 | + Thread.sleep( (long)(Math.random() * 100)); | ||
127 | + | ||
128 | + temp = temp + firstFactor; | ||
129 | + Thread.sleep( (long)(Math.random() * 100)); | ||
130 | + | ||
131 | + temp = temp + secondFactor; | ||
132 | + Thread.sleep( (long)(Math.random() * 100)); | ||
133 | + | ||
134 | + return temp; | ||
135 | + } catch (InterruptedException e) { | ||
136 | + throw new RuntimeException(e); | ||
137 | + } | ||
138 | + } | ||
139 | + | ||
140 | + @Operation | ||
141 | + public synchronized Integer calculateFactorsSynchronized(Integer firstFactor , Integer secondFactor){ | ||
142 | + setFirstFactor(firstFactor); | ||
143 | + setSecondFactor(secondFactor); | ||
144 | + | ||
145 | + try { | ||
146 | + int temp = firstFactor + secondFactor; | ||
147 | + Thread.sleep( (long)(Math.random() * 100)); | ||
148 | + | ||
149 | + temp = temp + firstFactor; | ||
150 | + Thread.sleep( (long)(Math.random() * 100)); | ||
151 | + | ||
152 | + temp = temp + secondFactor; | ||
153 | + Thread.sleep( (long)(Math.random() * 100)); | ||
154 | + | ||
155 | + return temp; | ||
156 | + } catch (InterruptedException e) { | ||
157 | + throw new RuntimeException(e); | ||
158 | + } | ||
159 | + } | ||
160 | + | ||
161 | + public void nonOperationAnnotatedMethod(){ | ||
162 | + System.out.println("Test"); | ||
163 | + } | ||
164 | + | ||
165 | + | ||
74 | } | 166 | } |
impl/core/src/test/java/management/testclasses/DummyManagementExtension.java
@@ -4,7 +4,6 @@ import java.util.List; | @@ -4,7 +4,6 @@ import java.util.List; | ||
4 | 4 | ||
5 | import javax.inject.Inject; | 5 | import javax.inject.Inject; |
6 | 6 | ||
7 | -import management.ManagedClassStore; | ||
8 | 7 | ||
9 | import br.gov.frameworkdemoiselle.management.extension.ManagementExtension; | 8 | import br.gov.frameworkdemoiselle.management.extension.ManagementExtension; |
10 | import br.gov.frameworkdemoiselle.management.internal.ManagedType; | 9 | import br.gov.frameworkdemoiselle.management.internal.ManagedType; |
impl/core/src/test/java/management/testclasses/ManagedClassStore.java
0 → 100644
@@ -0,0 +1,66 @@ | @@ -0,0 +1,66 @@ | ||
1 | +package management.testclasses; | ||
2 | + | ||
3 | +import java.util.ArrayList; | ||
4 | +import java.util.Collection; | ||
5 | +import java.util.List; | ||
6 | + | ||
7 | +import javax.enterprise.context.ApplicationScoped; | ||
8 | + | ||
9 | +import br.gov.frameworkdemoiselle.management.internal.ManagedType; | ||
10 | +import br.gov.frameworkdemoiselle.management.internal.MonitoringManager; | ||
11 | +import br.gov.frameworkdemoiselle.util.Beans; | ||
12 | + | ||
13 | +/** | ||
14 | + * Dummy class that stores managed types detected by the management bootstrap | ||
15 | + * and can read/write properties and invoke operations on them, simulating a management | ||
16 | + * extension like JMX or SNMP. | ||
17 | + * | ||
18 | + * @author serpro | ||
19 | + * | ||
20 | + */ | ||
21 | +@ApplicationScoped | ||
22 | +public class ManagedClassStore { | ||
23 | + | ||
24 | + private List<ManagedType> managedTypes = new ArrayList<ManagedType>(); | ||
25 | + | ||
26 | + | ||
27 | + public List<ManagedType> getManagedTypes() { | ||
28 | + return managedTypes; | ||
29 | + } | ||
30 | + | ||
31 | + public void addManagedTypes(Collection<ManagedType> managedTypes){ | ||
32 | + this.managedTypes.addAll(managedTypes); | ||
33 | + } | ||
34 | + | ||
35 | + public void setProperty(Class<?> managedClass , String attributeName , Object newValue){ | ||
36 | + MonitoringManager manager = Beans.getReference(MonitoringManager.class); | ||
37 | + for (ManagedType type : manager.getManagedTypes()){ | ||
38 | + if (type.getType().equals(managedClass)){ | ||
39 | + manager.setProperty(type, attributeName, newValue); | ||
40 | + break; | ||
41 | + } | ||
42 | + } | ||
43 | + } | ||
44 | + | ||
45 | + public Object getProperty(Class<?> managedClass , String attributeName ){ | ||
46 | + MonitoringManager manager = Beans.getReference(MonitoringManager.class); | ||
47 | + for (ManagedType type : manager.getManagedTypes()){ | ||
48 | + if (type.getType().equals(managedClass)){ | ||
49 | + return manager.getProperty(type, attributeName); | ||
50 | + } | ||
51 | + } | ||
52 | + | ||
53 | + return null; | ||
54 | + } | ||
55 | + | ||
56 | + public Object invoke(Class<?> managedClass , String operation , Object... params){ | ||
57 | + MonitoringManager manager = Beans.getReference(MonitoringManager.class); | ||
58 | + for (ManagedType type : manager.getManagedTypes()){ | ||
59 | + if (type.getType().equals(managedClass)){ | ||
60 | + return manager.invoke(type, operation, params); | ||
61 | + } | ||
62 | + } | ||
63 | + | ||
64 | + return null; | ||
65 | + } | ||
66 | +} |