aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAleksi Hietanen <aleksi@vaadin.com>2016-10-14 11:41:06 +0300
committerVaadin Code Review <review@vaadin.com>2016-10-24 07:55:31 +0000
commit2cdb3b39329232dcefee2ae61ded92f2c3fe54b0 (patch)
treee1bcdcab7065ae5b5723e8400580d50055aae77f
parent02ed73dc9ea247b13fbef63007af6c3c53ad9423 (diff)
downloadvaadin-framework-2cdb3b39329232dcefee2ae61ded92f2c3fe54b0.tar.gz
vaadin-framework-2cdb3b39329232dcefee2ae61ded92f2c3fe54b0.zip
Add utility for outputting the full declarative syntax of components
Change-Id: I4bc740154ffb5a30892b1859a7550a7aeff94fb3
-rw-r--r--compatibility-server/src/test/java/com/vaadin/tests/server/ClasspathHelper.java121
-rw-r--r--compatibility-server/src/test/java/com/vaadin/tests/server/ComponentDesignWriterUtility.java204
-rw-r--r--compatibility-server/src/test/java/com/vaadin/tests/server/DeprecatedTest.java107
-rw-r--r--compatibility-server/src/test/java/com/vaadin/v7/tests/VaadinClasses.java12
-rw-r--r--server/src/main/java/com/vaadin/ui/declarative/DesignAttributeHandler.java17
-rw-r--r--server/src/main/java/com/vaadin/ui/declarative/DesignContext.java3
6 files changed, 370 insertions, 94 deletions
diff --git a/compatibility-server/src/test/java/com/vaadin/tests/server/ClasspathHelper.java b/compatibility-server/src/test/java/com/vaadin/tests/server/ClasspathHelper.java
new file mode 100644
index 0000000000..d32718196b
--- /dev/null
+++ b/compatibility-server/src/test/java/com/vaadin/tests/server/ClasspathHelper.java
@@ -0,0 +1,121 @@
+package com.vaadin.tests.server;
+
+import java.io.File;
+import java.io.IOException;
+import java.lang.reflect.Modifier;
+import java.net.URI;
+import java.nio.file.FileSystems;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.List;
+import java.util.Locale;
+import java.util.Objects;
+import java.util.function.Predicate;
+import java.util.stream.Collectors;
+import java.util.stream.Stream;
+
+/**
+ * Allows to get classes from the current classpath using classes FQN filter.
+ * <p>
+ * The methods in the class return all real (not anonymous and not private)
+ * classes from the filtered classpath.
+ *
+ * @author Vaadin Ltd
+ *
+ */
+class ClasspathHelper {
+
+ private final Predicate<String> skipClassesFilter;
+
+ ClasspathHelper(Predicate<String> skipClassesFilter) {
+ this.skipClassesFilter = skipClassesFilter;
+ }
+
+ Stream<Class<?>> getVaadinClassesFromClasspath(
+ Predicate<String> classpathFilter,
+ Predicate<Class<?>> classFilter) {
+ return getRawClasspathEntries().stream().filter(classpathFilter)
+ .map(File::new).map(file -> getVaadinClassesFromFile(file))
+ .flatMap(List::stream).filter(classFilter)
+ .filter(cls -> !cls.isSynthetic() && !cls.isAnonymousClass()
+ && !Modifier.isPrivate(cls.getModifiers()));
+
+ }
+
+ Stream<Class<?>> getVaadinClassesFromClasspath(
+ Predicate<String> classpathFilter) {
+ return getVaadinClassesFromClasspath(classpathFilter, cls -> true);
+ }
+
+ private List<Class<?>> getVaadinClassesFromFile(File classesRoot) {
+ try {
+ if (classesRoot.isDirectory()) {
+ return Files.walk(classesRoot.toPath())
+ .filter(Files::isRegularFile)
+ .filter(path -> path.toFile().getName()
+ .endsWith(".class"))
+ .filter(path -> classesRoot.toPath().relativize(path)
+ .toString().contains("com/vaadin/"))
+ .map(path -> getClassFromFile(path,
+ classesRoot.toPath()))
+ .filter(Objects::nonNull).collect(Collectors.toList());
+ } else if (classesRoot.getName().toLowerCase(Locale.ENGLISH)
+ .endsWith(".jar")) {
+ URI uri = URI.create("jar:file:" + classesRoot.getPath());
+ Path root = FileSystems
+ .newFileSystem(uri, Collections.emptyMap())
+ .getPath("/");
+ return Files.walk(root).filter(Files::isRegularFile)
+ .filter(path -> path.toUri().getSchemeSpecificPart()
+ .endsWith(".class"))
+ .filter(path -> root.relativize(path).toString()
+ .contains("com/vaadin/"))
+ .map(path -> getClassFromFile(path, root))
+ .filter(Objects::nonNull).collect(Collectors.toList());
+ }
+ return null;
+ } catch (IOException e) {
+ throw new RuntimeException(e);
+ }
+ }
+
+ private Class<?> getClassFromFile(Path path, Path root) {
+ Path relative = root.relativize(path);
+ String name = relative.toString();
+ name = name.substring(0, name.length() - ".class".length());
+ name = name.replace('/', '.');
+ if (skipClassesFilter.test(name)) {
+ return null;
+ }
+ try {
+ return Class.forName(name, false, getClass().getClassLoader());
+ } catch (ClassNotFoundException e) {
+ throw new RuntimeException(e);
+ }
+ }
+
+ private final static List<String> getRawClasspathEntries() {
+ List<String> locations = new ArrayList<>();
+
+ String pathSep = System.getProperty("path.separator");
+ String classpath = System.getProperty("java.class.path");
+
+ if (classpath.startsWith("\"")) {
+ classpath = classpath.substring(1);
+ }
+ if (classpath.endsWith("\"")) {
+ classpath = classpath.substring(0, classpath.length() - 1);
+ }
+
+ String[] split = classpath.split(pathSep);
+ for (int i = 0; i < split.length; i++) {
+ String classpathEntry = split[i];
+ locations.add(classpathEntry);
+ }
+
+ return locations;
+ }
+
+}
diff --git a/compatibility-server/src/test/java/com/vaadin/tests/server/ComponentDesignWriterUtility.java b/compatibility-server/src/test/java/com/vaadin/tests/server/ComponentDesignWriterUtility.java
new file mode 100644
index 0000000000..15ae1c9cf4
--- /dev/null
+++ b/compatibility-server/src/test/java/com/vaadin/tests/server/ComponentDesignWriterUtility.java
@@ -0,0 +1,204 @@
+package com.vaadin.tests.server;
+
+import java.io.File;
+import java.lang.reflect.Modifier;
+import java.net.URISyntaxException;
+import java.util.Arrays;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+import java.util.stream.Collectors;
+
+import org.jsoup.nodes.Document;
+import org.jsoup.nodes.Element;
+import org.junit.Assert;
+import org.junit.Test;
+
+import com.vaadin.navigator.Navigator;
+import com.vaadin.ui.Component;
+import com.vaadin.ui.DragAndDropWrapper;
+import com.vaadin.ui.components.colorpicker.ColorPickerGradient;
+import com.vaadin.ui.components.colorpicker.ColorPickerGrid;
+import com.vaadin.ui.components.colorpicker.ColorPickerHistory;
+import com.vaadin.ui.components.colorpicker.ColorPickerPopup;
+import com.vaadin.ui.components.colorpicker.ColorPickerPreview;
+import com.vaadin.ui.components.colorpicker.ColorPickerSelect;
+import com.vaadin.ui.declarative.Design;
+import com.vaadin.ui.declarative.DesignAttributeHandler;
+import com.vaadin.ui.declarative.DesignContext;
+
+/**
+ * Utility class for outputting the declarative syntax of Vaadin components.
+ */
+public class ComponentDesignWriterUtility {
+
+ private static final Set<String> WHITE_LIST_FQNS = new HashSet<>();
+ static {
+ WHITE_LIST_FQNS.add(DragAndDropWrapper.class.getName());
+ WHITE_LIST_FQNS.add(Navigator.EmptyView.class.getName());
+
+ WHITE_LIST_FQNS.add(ColorPickerGradient.class.getName());
+ WHITE_LIST_FQNS.add(ColorPickerPopup.class.getName());
+ WHITE_LIST_FQNS.add(ColorPickerPreview.class.getName());
+ WHITE_LIST_FQNS.add(ColorPickerGrid.class.getName());
+ WHITE_LIST_FQNS.add(ColorPickerSelect.class.getName());
+ WHITE_LIST_FQNS.add(ColorPickerHistory.class.getName());
+
+ WHITE_LIST_FQNS
+ .add(com.vaadin.v7.ui.components.colorpicker.ColorPickerGradient.class
+ .getName());
+ WHITE_LIST_FQNS
+ .add(com.vaadin.v7.ui.components.colorpicker.ColorPickerPopup.class
+ .getName());
+ WHITE_LIST_FQNS
+ .add(com.vaadin.v7.ui.components.colorpicker.ColorPickerPreview.class
+ .getName());
+ WHITE_LIST_FQNS
+ .add(com.vaadin.v7.ui.components.colorpicker.ColorPickerGrid.class
+ .getName());
+ WHITE_LIST_FQNS
+ .add(com.vaadin.v7.ui.components.colorpicker.ColorPickerSelect.class
+ .getName());
+ WHITE_LIST_FQNS
+ .add(com.vaadin.v7.ui.components.colorpicker.ColorPickerHistory.class
+ .getName());
+
+ // ==================================================================
+ // Classes that cannot be loaded
+ // ==================================================================
+ WHITE_LIST_FQNS.add(
+ "com.vaadin.server.communication.PushAtmosphereHandler$AtmosphereResourceListener");
+ WHITE_LIST_FQNS
+ .add("com.vaadin.server.communication.PushAtmosphereHandler");
+ WHITE_LIST_FQNS
+ .add("com.vaadin.server.communication.PushRequestHandler$2");
+ WHITE_LIST_FQNS.add("com.vaadin.server.LegacyVaadinPortlet");
+ WHITE_LIST_FQNS.add("com.vaadin.server.RestrictedRenderResponse");
+ WHITE_LIST_FQNS
+ .add("com.vaadin.server.VaadinPortlet$VaadinGateInRequest");
+ WHITE_LIST_FQNS.add(
+ "com.vaadin.server.VaadinPortlet$VaadinHttpAndPortletRequest");
+ WHITE_LIST_FQNS
+ .add("com.vaadin.server.VaadinPortlet$VaadinLiferayRequest");
+ WHITE_LIST_FQNS.add(
+ "com.vaadin.server.VaadinPortlet$VaadinWebLogicPortalRequest");
+ WHITE_LIST_FQNS.add(
+ "com.vaadin.server.VaadinPortlet$VaadinWebSpherePortalRequest");
+ WHITE_LIST_FQNS.add("com.vaadin.server.VaadinPortlet");
+ WHITE_LIST_FQNS.add("com.vaadin.server.VaadinPortletRequest");
+ }
+
+ private static final Document document = new Document("");
+ private static final DesignContext designContext = new DesignContext(
+ document);
+
+ @SafeVarargs
+ public static List<String> getDeclarativeSyntax(
+ Class<? extends Component>... components) {
+ return getDeclarativeSyntax(Arrays.asList(components));
+ }
+
+ public static List<String> getDeclarativeSyntax(
+ List<Class<? extends Component>> components) {
+ DesignAttributeHandler.setWriteDefaultValues(true);
+
+ List<String> declarativeStrings = components.stream()
+ .map(ComponentDesignWriterUtility::getDeclarativeSyntax)
+ .collect(Collectors.toList());
+
+ DesignAttributeHandler.setWriteDefaultValues(false);
+ return declarativeStrings;
+ }
+
+ @Test
+ public void vaadin8ComponentsElementStartsWithVaadinPrefix()
+ throws URISyntaxException {
+ Assert.assertTrue(getVaadin8Components().stream()
+ .map(ComponentDesignWriterUtility::getDeclarativeSyntax)
+ .allMatch(element -> element.startsWith("<vaadin-")));
+ }
+
+ @Test
+ public void vaadin7ComponentsElementStartsWithVaadinPrefix()
+ throws URISyntaxException {
+ Assert.assertTrue(getVaadin7Components().stream()
+ .map(ComponentDesignWriterUtility::getDeclarativeSyntax)
+ .allMatch(element -> element.startsWith("<vaadin7-")));
+ }
+
+ private static String getDeclarativeSyntax(
+ Class<? extends Component> componentClass) {
+ try {
+ Component component = componentClass.newInstance();
+ Element element = document.createElement(Design.getComponentMapper()
+ .componentToTag(component, designContext));
+ component.writeDesign(element, designContext);
+ return element.toString();
+ } catch (Exception e) {
+ throw new RuntimeException(
+ "Could not write the declarative syntax for component "
+ + componentClass.getName(),
+ e);
+ }
+ }
+
+ public static void main(String[] args) throws URISyntaxException {
+ System.out.println("Vaadin 8 components:");
+ printFullDeclarativeSyntax(getVaadin8Components());
+
+ System.out.println("Vaadin 7 components:");
+ printFullDeclarativeSyntax(getVaadin7Components());
+
+ System.out.println("\nClases that are explicitely excluded from "
+ + "the design support introspection:");
+ WHITE_LIST_FQNS.forEach(System.out::println);
+ }
+
+ private static void printFullDeclarativeSyntax(
+ List<Class<? extends Component>> components) {
+ DesignAttributeHandler.setWriteDefaultValues(true);
+ components.stream().forEach(component -> System.out
+ .println(getDeclarativeSyntax(component)));
+ DesignAttributeHandler.setWriteDefaultValues(false);
+ }
+
+ private static List<Class<? extends Component>> getVaadin8Components()
+ throws URISyntaxException {
+ List<Class<? extends Component>> vaadin8Components = getVaadinComponentsFromClasspath(
+ "/server/target/classes");
+ if (vaadin8Components.isEmpty()) {
+ throw new RuntimeException(
+ "No vaadin 8 components found on your classpath.");
+ }
+ return vaadin8Components;
+ }
+
+ private static List<Class<? extends Component>> getVaadin7Components()
+ throws URISyntaxException {
+ List<Class<? extends Component>> vaadin7Components = getVaadinComponentsFromClasspath(
+ "compatibility-server");
+ if (vaadin7Components.isEmpty()) {
+ throw new RuntimeException(
+ "No vaadin 7 components found on your classpath.");
+ }
+ return vaadin7Components;
+ }
+
+ @SuppressWarnings("unchecked")
+ private static List<Class<? extends Component>> getVaadinComponentsFromClasspath(
+ String classpathFilter) throws URISyntaxException {
+ File testRoot = new File(
+ ComponentDesignWriterUtility.class.getResource("/").toURI());
+ List<Class<? extends Component>> classes = new ClasspathHelper(
+ WHITE_LIST_FQNS::contains).getVaadinClassesFromClasspath(
+ entry -> entry.contains(classpathFilter)
+ && !testRoot.equals(new File(entry)),
+ cls -> Component.class.isAssignableFrom(cls)
+ && !cls.isInterface()
+ && !Modifier.isAbstract(cls.getModifiers()))
+ .map(cls -> (Class<? extends Component>) cls)
+ .collect(Collectors.toList());
+ return classes;
+ }
+
+}
diff --git a/compatibility-server/src/test/java/com/vaadin/tests/server/DeprecatedTest.java b/compatibility-server/src/test/java/com/vaadin/tests/server/DeprecatedTest.java
index e8475b6750..671fc65080 100644
--- a/compatibility-server/src/test/java/com/vaadin/tests/server/DeprecatedTest.java
+++ b/compatibility-server/src/test/java/com/vaadin/tests/server/DeprecatedTest.java
@@ -16,19 +16,8 @@
package com.vaadin.tests.server;
import java.io.File;
-import java.io.IOException;
-import java.lang.reflect.Modifier;
-import java.net.URI;
import java.net.URISyntaxException;
-import java.net.URL;
-import java.nio.file.FileSystems;
-import java.nio.file.Files;
-import java.nio.file.Path;
-import java.util.ArrayList;
-import java.util.Collections;
-import java.util.List;
-import java.util.Locale;
-import java.util.stream.Collectors;
+import java.util.concurrent.atomic.AtomicInteger;
import org.junit.Assert;
import org.junit.Test;
@@ -39,86 +28,24 @@ import org.junit.Test;
*/
public class DeprecatedTest {
- private static String CLASS_SUFFIX = ".class";
-
@Test
public void allTypesAreDeprecated() throws URISyntaxException {
- URL url = DeprecatedTest.class.getResource("/");
- File file = new File(url.toURI());
- List<File> classpath = getRawClasspathEntries().stream()
- .filter(entry -> entry.contains("compatibility-server"))
- .map(File::new).filter(fileEntry -> !fileEntry.equals(file))
- .collect(Collectors.toList());
- Assert.assertFalse(classpath.isEmpty());
- classpath.forEach(this::checkDeprecatedClasses);
- }
-
- private void checkDeprecatedClasses(File classesRoot) {
- try {
- if (classesRoot.isDirectory()) {
- Files.walk(classesRoot.toPath()).filter(Files::isRegularFile)
- .filter(path -> path.toFile().getName()
- .endsWith(CLASS_SUFFIX))
- .forEach(path -> checkDeprecatedClass(path,
- classesRoot.toPath()));
- } else if (classesRoot.getName().toLowerCase(Locale.ENGLISH)
- .endsWith(".jar")) {
- URI uri = URI.create("jar:file:" + classesRoot.getPath());
- Path root = FileSystems
- .newFileSystem(uri, Collections.emptyMap())
- .getPath("/");
- Files.walk(root).filter(Files::isRegularFile)
- .filter(path -> path.toUri().getSchemeSpecificPart()
- .endsWith(CLASS_SUFFIX))
- .forEach(path -> checkDeprecatedClass(path, root));
- }
- } catch (IOException e) {
- throw new RuntimeException(e);
- }
- }
-
- private void checkDeprecatedClass(Path path, Path root) {
- Path relative = root.relativize(path);
- String name = relative.toString();
- name = name.substring(0, name.length() - CLASS_SUFFIX.length());
- name = name.replace('/', '.');
- try {
- Class<?> clazz = Class.forName(name);
- if (clazz.isSynthetic() || clazz.isAnonymousClass()) {
- return;
- }
- if (Modifier.isPrivate(clazz.getModifiers())) {
- return;
- }
- Assert.assertNotNull(
- "Class " + clazz
- + " is in compatability package and it's not deprecated",
- clazz.getAnnotation(Deprecated.class));
- } catch (ClassNotFoundException e) {
- throw new RuntimeException(e);
- }
+ AtomicInteger count = new AtomicInteger(0);
+
+ File testRoot = new File(DeprecatedTest.class.getResource("/").toURI());
+
+ new ClasspathHelper(fqn -> false)
+ .getVaadinClassesFromClasspath(
+ entry -> entry.contains("compatibility-server")
+ && !testRoot.equals(new File(entry)))
+ .forEach(cls -> {
+ count.incrementAndGet();
+ Assert.assertNotNull(
+ "Class " + cls
+ + " is in compatability package and it's not deprecated",
+ cls.getAnnotation(Deprecated.class));
+ });
+ Assert.assertTrue(count.get() > 0);
}
- private final static List<String> getRawClasspathEntries() {
- // try to keep the order of the classpath
- List<String> locations = new ArrayList<>();
-
- String pathSep = System.getProperty("path.separator");
- String classpath = System.getProperty("java.class.path");
-
- if (classpath.startsWith("\"")) {
- classpath = classpath.substring(1);
- }
- if (classpath.endsWith("\"")) {
- classpath = classpath.substring(0, classpath.length() - 1);
- }
-
- String[] split = classpath.split(pathSep);
- for (int i = 0; i < split.length; i++) {
- String classpathEntry = split[i];
- locations.add(classpathEntry);
- }
-
- return locations;
- }
}
diff --git a/compatibility-server/src/test/java/com/vaadin/v7/tests/VaadinClasses.java b/compatibility-server/src/test/java/com/vaadin/v7/tests/VaadinClasses.java
index b37f1bbe86..251f8b0223 100644
--- a/compatibility-server/src/test/java/com/vaadin/v7/tests/VaadinClasses.java
+++ b/compatibility-server/src/test/java/com/vaadin/v7/tests/VaadinClasses.java
@@ -3,6 +3,7 @@ package com.vaadin.v7.tests;
import java.io.IOException;
import java.util.List;
+import com.vaadin.ui.Component;
import com.vaadin.v7.ui.Field;
@SuppressWarnings("deprecation")
@@ -11,11 +12,20 @@ public class VaadinClasses {
public static List<Class<? extends Field>> getFields() {
try {
return com.vaadin.tests.VaadinClasses.findClasses(Field.class,
- "com.vaadin.ui");
+ "com.vaadin.v7.ui");
} catch (IOException e) {
e.printStackTrace();
return null;
}
}
+ public static List<Class<? extends Component>> getComponents() {
+ try {
+ return com.vaadin.tests.VaadinClasses.findClasses(Component.class,
+ "com.vaadin.v7.ui");
+ } catch (IOException e) {
+ throw new RuntimeException(
+ "Could not find all Vaadin component classes", e);
+ }
+ }
}
diff --git a/server/src/main/java/com/vaadin/ui/declarative/DesignAttributeHandler.java b/server/src/main/java/com/vaadin/ui/declarative/DesignAttributeHandler.java
index a70ab256e1..4290620051 100644
--- a/server/src/main/java/com/vaadin/ui/declarative/DesignAttributeHandler.java
+++ b/server/src/main/java/com/vaadin/ui/declarative/DesignAttributeHandler.java
@@ -60,6 +60,21 @@ public class DesignAttributeHandler implements Serializable {
// translates string <-> object
private static DesignFormatter FORMATTER = new DesignFormatter();
+ private static boolean writeDefaultValues = false;
+
+ /**
+ * Set whether default attribute values should be written by the
+ * {@code DesignAttributeHandler#writeAttribute(String, Attributes, Object, Object, Class)}
+ * method. Default is {@code false}.
+ *
+ * @param value
+ * {@code true} to write default values of attributes,
+ * {@code false} to disable writing of default values
+ */
+ public static void setWriteDefaultValues(boolean value) {
+ writeDefaultValues = value;
+ }
+
/**
* Returns the currently used formatter. All primitive types and all types
* needed by Vaadin components are handled by that formatter.
@@ -238,7 +253,7 @@ public class DesignAttributeHandler implements Serializable {
throw new IllegalArgumentException(
"input type: " + inputType.getName() + " not supported");
}
- if (!SharedUtil.equals(value, defaultValue)) {
+ if (writeDefaultValues || !SharedUtil.equals(value, defaultValue)) {
String attributeValue = toAttributeValue(inputType, value);
if ("".equals(attributeValue) && (inputType == boolean.class
|| inputType == Boolean.class)) {
diff --git a/server/src/main/java/com/vaadin/ui/declarative/DesignContext.java b/server/src/main/java/com/vaadin/ui/declarative/DesignContext.java
index 35145c45fb..ecbd5e638c 100644
--- a/server/src/main/java/com/vaadin/ui/declarative/DesignContext.java
+++ b/server/src/main/java/com/vaadin/ui/declarative/DesignContext.java
@@ -827,8 +827,7 @@ public class DesignContext implements Serializable {
String value) {
Map<String, String> map = customAttributes.get(component);
if (map == null) {
- customAttributes.put(component,
- map = new HashMap<>());
+ customAttributes.put(component, map = new HashMap<>());
}
map.put(attribute, value);
}