# HG changeset patch # User Mike Becker # Date 1589489281 -7200 # Node ID 57cfb94ab99fe561af0abb45757abe570bc49610 # Parent 1574965c7dc722629f219b36bca6c7a4787668f2 projects can now be added and updated diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/AbstractLightPITServlet.java --- a/src/main/java/de/uapcore/lightpit/AbstractLightPITServlet.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/AbstractLightPITServlet.java Thu May 14 22:48:01 2020 +0200 @@ -39,6 +39,7 @@ import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpSession; import java.io.IOException; +import java.lang.reflect.Constructor; import java.lang.reflect.Method; import java.lang.reflect.Modifier; import java.sql.Connection; @@ -225,6 +226,18 @@ } /** + * @param req the servlet request object + * @param location the location where to redirect + * @see Constants#REQ_ATTR_REDIRECT_LOCATION + */ + public void setRedirectLocation(HttpServletRequest req, String location) { + if (location.startsWith("./")) { + location = location.replaceFirst("\\./", Functions.baseHref(req)); + } + req.setAttribute(Constants.REQ_ATTR_REDIRECT_LOCATION, location); + } + + /** * Specifies the name of an additional stylesheet used by the module. *

* Setting an additional stylesheet is optional, but quite common for HTML @@ -240,6 +253,30 @@ req.setAttribute(Constants.REQ_ATTR_STYLESHEET, Functions.enforceExt(stylesheet, ".css")); } + /** + * Obtains a request parameter of the specified type. + * The specified type must have a single-argument constructor accepting a string to perform conversion. + * The constructor of the specified type may throw an exception on conversion failures. + * + * @param req the servlet request object + * @param clazz the class object of the expected type + * @param name the name of the parameter + * @param the expected type + * @return the parameter value or an empty optional, if no parameter with the specified name was found + */ + public Optional getParameter(HttpServletRequest req, Class clazz, String name) { + final String paramValue = req.getParameter(name); + if (paramValue == null) return Optional.empty(); + if (clazz.equals(String.class)) return Optional.of((T)paramValue); + try { + final Constructor ctor = clazz.getConstructor(String.class); + return Optional.of(ctor.newInstance(paramValue)); + } catch (ReflectiveOperationException e) { + throw new RuntimeException(e); + } + + } + private void forwardToFullView(HttpServletRequest req, HttpServletResponse resp) throws IOException, ServletException { @@ -287,6 +324,7 @@ } // set some internal request attributes + req.setAttribute(Constants.REQ_ATTR_BASE_HREF, Functions.baseHref(req)); req.setAttribute(Constants.REQ_ATTR_PATH, Functions.fullPath(req)); Optional.ofNullable(moduleInfo).ifPresent((proxy) -> req.setAttribute(Constants.REQ_ATTR_MODULE_INFO, proxy)); diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/Constants.java --- a/src/main/java/de/uapcore/lightpit/Constants.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/Constants.java Thu May 14 22:48:01 2020 +0200 @@ -40,6 +40,7 @@ public static final String DYN_FRAGMENT_PATH_PREFIX = "/WEB-INF/dynamic_fragments/"; + public static final String DYN_FRAGMENT_COMMIT_SUCCESSFUL = "commit-successful"; /** * Name for the context parameter specifying the available languages. @@ -77,6 +78,11 @@ public static final String REQ_ATTR_SUB_MENU = fqn(AbstractLightPITServlet.class, "subMenu"); /** + * Key for the request attribute containing the base href. + */ + public static final String REQ_ATTR_BASE_HREF = fqn(AbstractLightPITServlet.class, "base_href"); + + /** * Key for the request attribute containing the full path information (servlet path + path info). */ public static final String REQ_ATTR_PATH = fqn(AbstractLightPITServlet.class, "path"); @@ -91,6 +97,11 @@ */ public static final String REQ_ATTR_STYLESHEET = fqn(AbstractLightPITServlet.class, "extraCss"); + /** + * Key for a location the page shall redirect to. + * Will be used in a meta element. + */ + public static final String REQ_ATTR_REDIRECT_LOCATION = fqn(AbstractLightPITServlet.class, "redirectLocation"); /** * Key for the current language selection within the session. diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/Functions.java --- a/src/main/java/de/uapcore/lightpit/Functions.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/Functions.java Thu May 14 22:48:01 2020 +0200 @@ -66,15 +66,12 @@ return fqn(clazz.getName(), name); } - public static String fullPath(LightPITModule module, RequestMapping mapping) { - StringBuilder sb = new StringBuilder(); - sb.append(module.modulePath()); - sb.append('/'); - if (!mapping.requestPath().isEmpty()) { - sb.append(mapping.requestPath().isEmpty()); - sb.append('/'); - } - return sb.toString(); + public static String baseHref(HttpServletRequest req) { + return String.format("%s://%s:%d%s/", + req.getScheme(), + req.getServerName(), + req.getServerPort(), + req.getContextPath()); } public static String fullPath(HttpServletRequest req) { diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/dao/AbstractDao.java --- a/src/main/java/de/uapcore/lightpit/dao/AbstractDao.java Wed May 13 21:46:26 2020 +0200 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,96 +0,0 @@ -/* - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. - * - * Copyright 2018 Mike Becker. All rights reserved. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * - * 1. Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * - * 2. Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" - * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE - * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR - * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF - * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS - * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN - * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) - * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE - * POSSIBILITY OF SUCH DAMAGE. - * - */ -package de.uapcore.lightpit.dao; - -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Types; -import java.util.ArrayList; -import java.util.List; -import java.util.Optional; -import java.util.function.Function; - -public abstract class AbstractDao implements GenericDao { - - private final PreparedStatement listQuery; - - protected AbstractDao(PreparedStatement listQuery) { - this.listQuery = listQuery; - } - - public final T mapColumns(ResultSet result) throws SQLException { - return mapColumns(result, ""); - } - - public abstract T mapColumns(ResultSet result, String qualifier) throws SQLException; - - /** - * Qualifies a column label if an qualifier is specified. - * - * @param qualifier an optional qualifier - * @param label the column label - * @return the label, qualified if necessary - */ - protected final String qual(String qualifier, String label) { - if (qualifier == null || qualifier.isBlank()) { - return label; - } else { - return qualifier + "." + label; - } - } - - protected final void setStringOrNull(PreparedStatement stmt, int index, String str) throws SQLException { - if (str == null || str.isBlank()) { - stmt.setNull(index, Types.VARCHAR); - } else { - stmt.setString(index, str); - } - } - - protected final void setForeignKeyOrNull(PreparedStatement stmt, int index, T instance, Function keyGetter) throws SQLException { - Integer key = Optional.ofNullable(instance).map(keyGetter).orElse(null); - if (key == null) { - stmt.setNull(index, Types.INTEGER); - } else { - stmt.setInt(index, key); - } - } - - @Override - public List list() throws SQLException { - List list = new ArrayList<>(); - try (ResultSet result = listQuery.executeQuery()) { - while (result.next()) { - list.add(mapColumns(result)); - } - } - return list; - } -} diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/dao/Functions.java --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/java/de/uapcore/lightpit/dao/Functions.java Thu May 14 22:48:01 2020 +0200 @@ -0,0 +1,62 @@ +/* + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + * + * Copyright 2018 Mike Becker. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + */ +package de.uapcore.lightpit.dao; + +import java.sql.PreparedStatement; +import java.sql.SQLException; +import java.sql.Types; +import java.util.Optional; +import java.util.function.Function; + +/** + * Some DAO utilities. + */ +public final class Functions { + + public static void setStringOrNull(PreparedStatement stmt, int index, String str) throws SQLException { + if (str == null || str.isBlank()) { + stmt.setNull(index, Types.VARCHAR); + } else { + stmt.setString(index, str); + } + } + + public static void setForeignKeyOrNull(PreparedStatement stmt, int index, T instance, Function keyGetter) throws SQLException { + Integer key = Optional.ofNullable(instance).map(keyGetter).orElse(null); + if (key == null) { + stmt.setNull(index, Types.INTEGER); + } else { + stmt.setInt(index, key); + } + } + + private Functions() { + + } +} diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/dao/GenericDao.java --- a/src/main/java/de/uapcore/lightpit/dao/GenericDao.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/dao/GenericDao.java Thu May 14 22:48:01 2020 +0200 @@ -41,6 +41,15 @@ List list() throws SQLException; /** + * Finds an entity by its integer ID. + * + * @param id the id + * @return the enity or null if there is no such entity + * @throws SQLException on any kind of SQL errors + */ + T find(int id) throws SQLException; + + /** * Inserts an instance into database. * * @param instance the instance to insert diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/dao/postgres/PGDataAccessObjects.java --- a/src/main/java/de/uapcore/lightpit/dao/postgres/PGDataAccessObjects.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/dao/postgres/PGDataAccessObjects.java Thu May 14 22:48:01 2020 +0200 @@ -41,9 +41,8 @@ private final ProjectDao projectDao; public PGDataAccessObjects(Connection connection) throws SQLException { - final PGUserDao pgUserDao = new PGUserDao(connection); - userDao = pgUserDao; - projectDao = new PGProjectDao(connection, pgUserDao); + userDao = new PGUserDao(connection); + projectDao = new PGProjectDao(connection); } @Override diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/dao/postgres/PGProjectDao.java --- a/src/main/java/de/uapcore/lightpit/dao/postgres/PGProjectDao.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/dao/postgres/PGProjectDao.java Thu May 14 22:48:01 2020 +0200 @@ -28,7 +28,7 @@ */ package de.uapcore.lightpit.dao.postgres; -import de.uapcore.lightpit.dao.AbstractDao; +import de.uapcore.lightpit.dao.GenericDao; import de.uapcore.lightpit.dao.ProjectDao; import de.uapcore.lightpit.entities.Project; import de.uapcore.lightpit.entities.User; @@ -37,18 +37,31 @@ import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; +import java.util.ArrayList; +import java.util.List; import java.util.Objects; -public final class PGProjectDao extends AbstractDao implements ProjectDao { +import static de.uapcore.lightpit.dao.Functions.setForeignKeyOrNull; +import static de.uapcore.lightpit.dao.Functions.setStringOrNull; - private final PGUserDao userDao; +public final class PGProjectDao implements ProjectDao, GenericDao { - private final PreparedStatement insert; - private final PreparedStatement update; + private final PreparedStatement insert, update, list, find; - public PGProjectDao(Connection connection, PGUserDao userDao) throws SQLException { - super(connection.prepareStatement( - "select * from lpit_project join lpit_user owner on lpit_project.owner = owner.userid")); + public PGProjectDao(Connection connection) throws SQLException { + list = connection.prepareStatement( + "select id, name, description, repourl, " + + "userid, username, lastname, givenname, mail " + + "from lpit_project " + + "left join lpit_user owner on lpit_project.owner = owner.userid " + + "order by name"); + + find = connection.prepareStatement( + "select id, name, description, repourl, " + + "userid, username, lastname, givenname, mail " + + "from lpit_project " + + "left join lpit_user owner on lpit_project.owner = owner.userid " + + "where id = ?"); insert = connection.prepareStatement( "insert into lpit_project (name, description, repourl, owner) values (?, ?, ?, ?)" @@ -56,17 +69,24 @@ update = connection.prepareStatement( "update lpit_project set name = ?, description = ?, repourl = ?, owner = ? where id = ?" ); - - this.userDao = userDao; } - @Override - public Project mapColumns(ResultSet result, String q) throws SQLException { - final var proj = new Project(result.getInt(qual(q, "id"))); - proj.setName(result.getString(qual(q, "name"))); - proj.setDescription(result.getString(qual(q, "description"))); - proj.setRepoUrl(result.getString(qual(q, "repourl"))); - proj.setOwner(userDao.mapColumns(result, "owner")); + public Project mapColumns(ResultSet result) throws SQLException { + final var proj = new Project(result.getInt("id")); + proj.setName(result.getString("name")); + proj.setDescription(result.getString("description")); + proj.setRepoUrl(result.getString("repourl")); + + final int id = result.getInt("userid"); + if (id != 0) { + final var user = new User(id); + user.setUsername(result.getString("username")); + user.setGivenname(result.getString("givenname")); + user.setLastname(result.getString("lastname")); + user.setMail(result.getString("mail")); + proj.setOwner(user); + } + return proj; } @@ -87,6 +107,30 @@ setStringOrNull(update, 2, instance.getDescription()); setStringOrNull(update, 3, instance.getRepoUrl()); setForeignKeyOrNull(update, 4, instance.getOwner(), User::getUserID); + update.setInt(5, instance.getId()); return update.executeUpdate() > 0; } + + @Override + public List list() throws SQLException { + List projects = new ArrayList<>(); + try (var result = list.executeQuery()) { + while (result.next()) { + projects.add(mapColumns(result)); + } + } + return projects; + } + + @Override + public Project find(int id) throws SQLException { + find.setInt(1, id); + try (var result = find.executeQuery()) { + if (result.next()) { + return mapColumns(result); + } else { + return null; + } + } + } } diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/dao/postgres/PGUserDao.java --- a/src/main/java/de/uapcore/lightpit/dao/postgres/PGUserDao.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/dao/postgres/PGUserDao.java Thu May 14 22:48:01 2020 +0200 @@ -28,7 +28,7 @@ */ package de.uapcore.lightpit.dao.postgres; -import de.uapcore.lightpit.dao.AbstractDao; +import de.uapcore.lightpit.dao.GenericDao; import de.uapcore.lightpit.dao.UserDao; import de.uapcore.lightpit.entities.User; @@ -36,26 +36,41 @@ import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; +import java.util.ArrayList; +import java.util.List; import java.util.Objects; -public final class PGUserDao extends AbstractDao implements UserDao { +import static de.uapcore.lightpit.dao.Functions.setStringOrNull; - private final PreparedStatement insert; - private final PreparedStatement update; +public final class PGUserDao implements UserDao, GenericDao { + + public static final String[] COLUMNS = { + "id", "username", "lastname", "givenname", "mail" + }; + + private final PreparedStatement insert, update, list, find; public PGUserDao(Connection connection) throws SQLException { - super(connection.prepareStatement("select * from lpit_user where userid >= 0 order by username")); + list = connection.prepareStatement( + "select userid, username, lastname, givenname, mail " + + "from lpit_user where userid >= 0 " + + "order by username"); + find = connection.prepareStatement( + "select userid, username, lastname, givenname, mail " + + "from lpit_user where userid = ? "); insert = connection.prepareStatement("insert into lpit_user (username, lastname, givenname, mail) values (?, ?, ?, ?)"); update = connection.prepareStatement("update lpit_user set lastname = ?, givenname = ?, mail = ? where userid = ?"); } - @Override - public User mapColumns(ResultSet result, String q) throws SQLException { - final var user = new User(result.getInt(qual(q, "userid"))); - user.setUsername(result.getString(qual(q, "username"))); - user.setGivenname(result.getString(qual(q, "givenname"))); - user.setLastname(result.getString(qual(q, "lastname"))); + public User mapColumns(ResultSet result) throws SQLException { + final int id = result.getInt("userid"); + if (id == 0) return null; + final var user = new User(id); + user.setUsername(result.getString("username")); + user.setGivenname(result.getString("givenname")); + user.setLastname(result.getString("lastname")); + user.setMail(result.getString("mail")); return user; } @@ -77,4 +92,27 @@ update.setInt(4, instance.getUserID()); return update.executeUpdate() > 0; } + + @Override + public List list() throws SQLException { + List users = new ArrayList<>(); + try (var result = list.executeQuery()) { + while (result.next()) { + users.add(mapColumns(result)); + } + } + return users; + } + + @Override + public User find(int id) throws SQLException { + find.setInt(1, id); + try (var result = find.executeQuery()) { + if (result.next()) { + return mapColumns(result); + } else { + return null; + } + } + } } diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/entities/Project.java --- a/src/main/java/de/uapcore/lightpit/entities/Project.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/entities/Project.java Thu May 14 22:48:01 2020 +0200 @@ -28,8 +28,6 @@ */ package de.uapcore.lightpit.entities; -import java.util.ArrayList; -import java.util.List; import java.util.Objects; public class Project { @@ -40,8 +38,6 @@ private String repoUrl; private User owner; - private final List versions = new ArrayList<>(); - public Project(int id) { this.id = id; } @@ -82,10 +78,6 @@ this.owner = owner; } - public List getVersions() { - return versions; - } - @Override public boolean equals(Object o) { if (this == o) return true; diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/entities/User.java --- a/src/main/java/de/uapcore/lightpit/entities/User.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/entities/User.java Thu May 14 22:48:01 2020 +0200 @@ -80,6 +80,19 @@ this.lastname = lastname; } + public String getDisplayname() { + StringBuilder dn = new StringBuilder(); + dn.append(givenname); + dn.append(' '); + dn.append(lastname); + dn.append(' '); + if (mail != null && !mail.isBlank()) { + dn.append("<"+mail+">"); + } + final var str = dn.toString().trim(); + return str.isBlank() ? username : str; + } + @Override public boolean equals(Object o) { if (this == o) return true; diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/modules/HomeModule.java --- a/src/main/java/de/uapcore/lightpit/modules/HomeModule.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/modules/HomeModule.java Thu May 14 22:48:01 2020 +0200 @@ -31,6 +31,7 @@ import de.uapcore.lightpit.*; import javax.servlet.annotation.WebServlet; +import javax.servlet.http.HttpServletRequest; /** * Entry point for the application. @@ -47,7 +48,10 @@ public final class HomeModule extends AbstractLightPITServlet { @RequestMapping(method = HttpMethod.GET) - public ResponseType handle() { + public ResponseType handle(HttpServletRequest req) { + + setDynamicFragment(req, "home"); + setStylesheet(req, "home"); return ResponseType.HTML; } diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/java/de/uapcore/lightpit/modules/ProjectsModule.java --- a/src/main/java/de/uapcore/lightpit/modules/ProjectsModule.java Wed May 13 21:46:26 2020 +0200 +++ b/src/main/java/de/uapcore/lightpit/modules/ProjectsModule.java Thu May 14 22:48:01 2020 +0200 @@ -31,9 +31,13 @@ import de.uapcore.lightpit.*; import de.uapcore.lightpit.dao.DataAccessObjects; +import de.uapcore.lightpit.entities.Project; +import de.uapcore.lightpit.entities.User; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServletRequest; +import java.sql.SQLException; +import java.util.Optional; @LightPITModule( bundleBaseName = "localization.projects", @@ -47,12 +51,59 @@ public final class ProjectsModule extends AbstractLightPITServlet { @RequestMapping(method = HttpMethod.GET) - public ResponseType index(HttpServletRequest req, DataAccessObjects dao) { + public ResponseType index(HttpServletRequest req, DataAccessObjects dao) throws SQLException { + final var projectDao = dao.getProjectDao(); + + req.setAttribute("projects", projectDao.list()); + setDynamicFragment(req, "projects"); return ResponseType.HTML; } - @RequestMapping(method = HttpMethod.GET, requestPath = "versions", menuKey = "menu.versions") + @RequestMapping(requestPath = "edit", method = HttpMethod.GET) + public ResponseType displayCreateForm(HttpServletRequest req, DataAccessObjects dao) throws SQLException { + final var projectDao = dao.getProjectDao(); + + Optional id = getParameter(req, Integer.class, "id"); + if (id.isPresent()) { + req.setAttribute("project", Optional.ofNullable(projectDao.find(id.get())).orElse(new Project(-1))); + } else { + req.setAttribute("project", new Project(-1)); + } + + setDynamicFragment(req, "project-form"); + + return ResponseType.HTML; + } + + @RequestMapping(requestPath = "commit", method = HttpMethod.POST) + public ResponseType commit(HttpServletRequest req, DataAccessObjects dao) { + + Project project = new Project(-1); + try { + project = new Project(getParameter(req, Integer.class, "id").orElseThrow()); + project.setName(getParameter(req, String.class, "name").orElseThrow()); + getParameter(req, String.class, "description").ifPresent(project::setDescription); + getParameter(req, String.class, "repoUrl").ifPresent(project::setRepoUrl); + getParameter(req, Integer.class, "owner").map( + ownerId -> ownerId >= 0 ? new User(ownerId) : null + ).ifPresent(project::setOwner); + + dao.getProjectDao().saveOrUpdate(project); + + setRedirectLocation(req, "./projects/"); + setDynamicFragment(req, Constants.DYN_FRAGMENT_COMMIT_SUCCESSFUL); + } catch (NullPointerException | NumberFormatException | SQLException ex) { + // TODO: set request attribute with error text + req.setAttribute("project", project); + setDynamicFragment(req, "project-form"); + } + + return ResponseType.HTML; + } + + + @RequestMapping(requestPath = "versions", method = HttpMethod.GET, menuKey = "menu.versions") public ResponseType versions(HttpServletRequest req, DataAccessObjects dao) { return ResponseType.HTML; diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/resources/localization/home.properties --- a/src/main/resources/localization/home.properties Wed May 13 21:46:26 2020 +0200 +++ b/src/main/resources/localization/home.properties Thu May 14 22:48:01 2020 +0200 @@ -24,3 +24,5 @@ name = Home Page description = The default page that is displayed when visiting the site. menuLabel = Home + +version=LightPIT - Version 0.1 (Snapshot) \ No newline at end of file diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/resources/localization/home_de.properties --- a/src/main/resources/localization/home_de.properties Wed May 13 21:46:26 2020 +0200 +++ b/src/main/resources/localization/home_de.properties Thu May 14 22:48:01 2020 +0200 @@ -24,3 +24,5 @@ name = Startseite description = Die Seite, die dem Benutzer standardm\u00e4\u00dfig beim Besuch angezeigt wird. menuLabel = Startseite + +version=LightPIT - Version 0.1 (Entwicklungsversion) diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/resources/localization/lightpit.properties --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/resources/localization/lightpit.properties Thu May 14 22:48:01 2020 +0200 @@ -0,0 +1,28 @@ +# Copyright 2018 Mike Becker. All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# 1. Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# +# 2. Redistributions in binary form must reproduce the above copyright +# notice, this list of conditions and the following disclaimer in the +# documentation and/or other materials provided with the distribution. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +button.okay=OK +button.cancel=Cancel + +commit.success=Operation successful - you will be redirected in a second. +commit.redirect-link=If redirection does not work, click the following link: \ No newline at end of file diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/resources/localization/lightpit_de.properties --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/resources/localization/lightpit_de.properties Thu May 14 22:48:01 2020 +0200 @@ -0,0 +1,28 @@ +# Copyright 2018 Mike Becker. All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# 1. Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# +# 2. Redistributions in binary form must reproduce the above copyright +# notice, this list of conditions and the following disclaimer in the +# documentation and/or other materials provided with the distribution. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +button.okay=OK +button.cancel=Abbrechen + +commit.success=Operation erfolgreich - Sie werden jeden Moment weitergeleitet. +commit.redirect-link=Falls die Weiterleitung nicht klappt, klicken Sie bitte hier: diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/resources/localization/projects.properties --- a/src/main/resources/localization/projects.properties Wed May 13 21:46:26 2020 +0200 +++ b/src/main/resources/localization/projects.properties Thu May 14 22:48:01 2020 +0200 @@ -20,7 +20,20 @@ # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + name=Project Management description=Allows the configuration of projects. menuLabel=Projects + menu.versions=Versions + +button.create=New Project + +no-projects=Welcome to LightPIT. Start off by creating a new project! + +thead.name=Name +thead.description=Description +thead.repoUrl=Repository +thead.owner=Project Lead + +placeholder.null-owner=Unassigned diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/resources/localization/projects_de.properties --- a/src/main/resources/localization/projects_de.properties Wed May 13 21:46:26 2020 +0200 +++ b/src/main/resources/localization/projects_de.properties Thu May 14 22:48:01 2020 +0200 @@ -20,7 +20,20 @@ # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + name=Projektverwaltung description=Erlaubt die Konfiguration von Projekten. menuLabel=Projekte -menu.versions=Versionen \ No newline at end of file + +menu.versions=Versionen + +button.create=Neues Projekt + +no-projects=Wilkommen bei LightPIT. Beginnen Sie mit der Erstellung eines Projektes! + +thead.name=Name +thead.description=Beschreibung +thead.repoUrl=Repository +thead.owner=Projektleitung + +placeholder.null-owner=Nicht Zugewiesen diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/webapp/WEB-INF/dynamic_fragments/commit-successful.jsp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/webapp/WEB-INF/dynamic_fragments/commit-successful.jsp Thu May 14 22:48:01 2020 +0200 @@ -0,0 +1,34 @@ +<%-- +DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + +Copyright 2018 Mike Becker. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + +1. Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + +2. Redistributions in binary form must reproduce the above copyright +notice, this list of conditions and the following disclaimer in the +documentation and/or other materials provided with the distribution. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +--%> +<%@page pageEncoding="UTF-8" %> +<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %> +<%@taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> + + + + + diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/webapp/WEB-INF/dynamic_fragments/error.jsp --- a/src/main/webapp/WEB-INF/dynamic_fragments/error.jsp Wed May 13 21:46:26 2020 +0200 +++ b/src/main/webapp/WEB-INF/dynamic_fragments/error.jsp Thu May 14 22:48:01 2020 +0200 @@ -25,11 +25,13 @@ OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. --%> <%@page pageEncoding="UTF-8" %> +<%@page import="de.uapcore.lightpit.Constants" %> <%@page import="de.uapcore.lightpit.modules.ErrorModule" %> <%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %> <%@taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> <%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %> + diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/webapp/WEB-INF/dynamic_fragments/home.jsp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/webapp/WEB-INF/dynamic_fragments/home.jsp Thu May 14 22:48:01 2020 +0200 @@ -0,0 +1,32 @@ +<%-- +DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + +Copyright 2018 Mike Becker. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + +1. Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + +2. Redistributions in binary form must reproduce the above copyright +notice, this list of conditions and the following disclaimer in the +documentation and/or other materials provided with the distribution. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +--%> +<%@page pageEncoding="UTF-8" %> +<%@taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> + +

+ +
\ No newline at end of file diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/webapp/WEB-INF/dynamic_fragments/project-form.jsp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/webapp/WEB-INF/dynamic_fragments/project-form.jsp Thu May 14 22:48:01 2020 +0200 @@ -0,0 +1,75 @@ +<%-- +DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + +Copyright 2018 Mike Becker. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + +1. Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + +2. Redistributions in binary form must reproduce the above copyright +notice, this list of conditions and the following disclaimer in the +documentation and/or other materials provided with the distribution. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +--%> +<%@page pageEncoding="UTF-8" %> +<%@page import="de.uapcore.lightpit.Constants" %> +<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %> +<%@taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> + + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + +
+
diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/webapp/WEB-INF/dynamic_fragments/projects.jsp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/webapp/WEB-INF/dynamic_fragments/projects.jsp Thu May 14 22:48:01 2020 +0200 @@ -0,0 +1,83 @@ +<%-- +DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + +Copyright 2018 Mike Becker. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + +1. Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + +2. Redistributions in binary form must reproduce the above copyright +notice, this list of conditions and the following disclaimer in the +documentation and/or other materials provided with the distribution. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +--%> +<%@page pageEncoding="UTF-8" %> +<%@page import="de.uapcore.lightpit.Constants" %> +<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %> +<%@taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> + + + + + + +
+ +
+
+ +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + "> + + + + +
+
diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/webapp/WEB-INF/jsp/site.jsp --- a/src/main/webapp/WEB-INF/jsp/site.jsp Wed May 13 21:46:26 2020 +0200 +++ b/src/main/webapp/WEB-INF/jsp/site.jsp Thu May 14 22:48:01 2020 +0200 @@ -31,7 +31,7 @@ <%@taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %> <%-- Make the base href easily available at request scope --%> - + <%-- Define an alias for the request path --%> @@ -45,6 +45,9 @@ <%-- Define an alias for the fragment name --%> +<%-- Define an alias for the optional redirect location --%> + + <%-- Define an alias for the additional stylesheet --%> @@ -66,6 +69,9 @@ + + + @@ -87,6 +93,7 @@
+
diff -r 1574965c7dc7 -r 57cfb94ab99f src/main/webapp/lightpit.css --- a/src/main/webapp/lightpit.css Wed May 13 21:46:26 2020 +0200 +++ b/src/main/webapp/lightpit.css Thu May 14 22:48:01 2020 +0200 @@ -28,22 +28,17 @@ */ html { - background: #f8f8f8; + font-family: sans-serif; + background: white; + color: #1c204e; + margin: 0; + padding: 0; } body { - background: white; - font-family: serif; - - border-color: #505050; - border-style: solid; - border-width: 1pt; - - color: #1c202e; -} - -h1, h2, h3, h4, #mainMenu, #subMenu { - font-family: sans-serif; + height: 100%; + margin: 0; + padding: 0; } a { @@ -88,6 +83,33 @@ padding: 1em; } +button, a.button { + display: inline-block; + font-size: medium; + border-style: outset; + border-width: 2pt; + border-color: #6060cc; + color: inherit; + background: #f0f0f0; + + padding: .25em .5em .25em .5em; + cursor: default; + text-decoration: none; +} + +button:hover, a.button:hover { + background: #f0f0ff; +} + +button[type=submit] { + background: #20a0ff; + color: white; +} + +button[type=submit]:hover { + background: #1090cf; +} + th { text-align: left; } @@ -101,6 +123,7 @@ } table.datatable th { + white-space: nowrap; font-weight: bold; background: lightsteelblue; } @@ -116,6 +139,35 @@ background: lightblue; } +table.formtable { + border-style: none; + border-collapse: separate; + border-spacing: 1em; +} + +table.formtable th { + font-weight: bold; + text-align: left; + vertical-align: center; + white-space: nowrap; +} + +table.formtable tbody td > * { + width: 100%; +} + +table.formtable tfoot td { + text-align: right; +} + +.fullwidth { + width: 100%; +} + +.vtop { + vertical-align: top; +} + .hcenter { text-align: center; } @@ -126,4 +178,16 @@ .nowrap { white-space: nowrap; +} + +.medskip { + margin-top: .5em; +} + +.info-box { + margin: 2em; + border-style: dashed; + border-width: 1pt; + border-color: deepskyblue; + padding: 1em; } \ No newline at end of file