NIHVIVO-2279 First stab at UserAccount and UserAccountsSelector, etc.
This commit is contained in:
parent
4a082abdc8
commit
78f966eb0f
8 changed files with 1100 additions and 0 deletions
|
@ -0,0 +1,112 @@
|
|||
/* $This file is distributed under the terms of the license in /doc/license.txt$ */
|
||||
|
||||
package edu.cornell.mannlib.vitro.webapp.beans;
|
||||
|
||||
import java.util.Collection;
|
||||
import java.util.Collections;
|
||||
import java.util.HashSet;
|
||||
import java.util.Set;
|
||||
|
||||
/**
|
||||
* Information about the account of a user. URI, email, password, etc.
|
||||
*/
|
||||
public class UserAccount {
|
||||
public enum Status {
|
||||
ACTIVE, INACTIVE
|
||||
}
|
||||
|
||||
private String uri = "";
|
||||
|
||||
private String emailAddress = "";
|
||||
private String firstName = "";
|
||||
private String lastName = "";
|
||||
|
||||
private String md5password = "";
|
||||
private long passwordChangeExpires = 0L;
|
||||
private int loginCount = 0;
|
||||
private Status status = Status.INACTIVE;
|
||||
|
||||
private Set<String> permissionSetUris = Collections.emptySet();
|
||||
|
||||
public String getUri() {
|
||||
return uri;
|
||||
}
|
||||
|
||||
public void setUri(String uri) {
|
||||
this.uri = uri;
|
||||
}
|
||||
|
||||
public String getEmailAddress() {
|
||||
return emailAddress;
|
||||
}
|
||||
|
||||
public void setEmailAddress(String emailAddress) {
|
||||
this.emailAddress = emailAddress;
|
||||
}
|
||||
|
||||
public String getFirstName() {
|
||||
return firstName;
|
||||
}
|
||||
|
||||
public void setFirstName(String firstName) {
|
||||
this.firstName = firstName;
|
||||
}
|
||||
|
||||
public String getLastName() {
|
||||
return lastName;
|
||||
}
|
||||
|
||||
public void setLastName(String lastName) {
|
||||
this.lastName = lastName;
|
||||
}
|
||||
|
||||
public String getMd5password() {
|
||||
return md5password;
|
||||
}
|
||||
|
||||
public void setMd5password(String md5password) {
|
||||
this.md5password = md5password;
|
||||
}
|
||||
|
||||
public long getPasswordChangeExpires() {
|
||||
return passwordChangeExpires;
|
||||
}
|
||||
|
||||
public void setPasswordChangeExpires(long passwordChangeExpires) {
|
||||
this.passwordChangeExpires = passwordChangeExpires;
|
||||
}
|
||||
|
||||
public int getLoginCount() {
|
||||
return loginCount;
|
||||
}
|
||||
|
||||
public void setLoginCount(int loginCount) {
|
||||
this.loginCount = loginCount;
|
||||
}
|
||||
|
||||
public Status getStatus() {
|
||||
return status;
|
||||
}
|
||||
|
||||
public void setStatus(Status status) {
|
||||
this.status = status;
|
||||
}
|
||||
|
||||
public Set<String> getPermissionSetUris() {
|
||||
return new HashSet<String>(permissionSetUris);
|
||||
}
|
||||
|
||||
public void setPermissionSetUris(Collection<String> permissionSetUris) {
|
||||
this.permissionSetUris = new HashSet<String>(permissionSetUris);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "UserAccount[uri=" + uri + ", emailAddress=" + emailAddress
|
||||
+ ", firstName=" + firstName + ", lastName=" + lastName
|
||||
+ ", md5password=" + md5password + ", passwordChangeExpires="
|
||||
+ passwordChangeExpires + ", loginCount=" + loginCount
|
||||
+ ", status=" + status + ", permissionSetUris="
|
||||
+ permissionSetUris + "]";
|
||||
}
|
||||
}
|
|
@ -0,0 +1,48 @@
|
|||
/* $This file is distributed under the terms of the license in /doc/license.txt$ */
|
||||
|
||||
package edu.cornell.mannlib.vitro.webapp.controller.freemarker.accountmanagement;
|
||||
|
||||
/**
|
||||
* How are the accounts to be sorted?
|
||||
*/
|
||||
public class UserAccountsOrdering {
|
||||
public enum Direction {
|
||||
ASCENDING("ASC"), DESCENDING("DESC");
|
||||
|
||||
public final String keyword;
|
||||
|
||||
Direction(String keyword) {
|
||||
this.keyword = keyword;
|
||||
}
|
||||
}
|
||||
|
||||
public enum Field {
|
||||
EMAIL("email"), FIRST_NAME("firstName"), LAST_NAME("lastName"), STATUS(
|
||||
"status"), ROLE("ps"), LOGIN_COUNT("count");
|
||||
|
||||
public final String variableName;
|
||||
|
||||
Field(String variableName) {
|
||||
this.variableName = variableName;
|
||||
}
|
||||
}
|
||||
|
||||
public static final UserAccountsOrdering DEFAULT_ORDERING = new UserAccountsOrdering(
|
||||
Field.EMAIL, Direction.ASCENDING);
|
||||
|
||||
private final Field field;
|
||||
private final Direction direction;
|
||||
|
||||
public UserAccountsOrdering(Field field, Direction direction) {
|
||||
this.field = field;
|
||||
this.direction = direction;
|
||||
}
|
||||
|
||||
public Field getField() {
|
||||
return field;
|
||||
}
|
||||
|
||||
public Direction getDirection() {
|
||||
return direction;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,40 @@
|
|||
/* $This file is distributed under the terms of the license in /doc/license.txt$ */
|
||||
|
||||
package edu.cornell.mannlib.vitro.webapp.controller.freemarker.accountmanagement;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collection;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
|
||||
import edu.cornell.mannlib.vitro.webapp.beans.UserAccount;
|
||||
|
||||
/**
|
||||
* A group of accounts (might be empty), with the criteria that were used to
|
||||
* select them.
|
||||
*/
|
||||
public class UserAccountsSelection {
|
||||
private final UserAccountsSelectionCriteria criteria;
|
||||
private final List<UserAccount> userAccounts;
|
||||
private final int resultCount;
|
||||
|
||||
public UserAccountsSelection(UserAccountsSelectionCriteria criteria,
|
||||
Collection<UserAccount> userAccounts, int resultCount) {
|
||||
this.criteria = criteria;
|
||||
this.userAccounts = Collections
|
||||
.unmodifiableList(new ArrayList<UserAccount>(userAccounts));
|
||||
this.resultCount = resultCount;
|
||||
}
|
||||
|
||||
public UserAccountsSelectionCriteria getCriteria() {
|
||||
return criteria;
|
||||
}
|
||||
|
||||
public List<UserAccount> getUserAccounts() {
|
||||
return userAccounts;
|
||||
}
|
||||
|
||||
public int getResultCount() {
|
||||
return resultCount;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,68 @@
|
|||
/* $This file is distributed under the terms of the license in /doc/license.txt$ */
|
||||
|
||||
package edu.cornell.mannlib.vitro.webapp.controller.freemarker.accountmanagement;
|
||||
|
||||
/**
|
||||
* On what basis are we selecting user accounts?
|
||||
*/
|
||||
public class UserAccountsSelectionCriteria {
|
||||
/** How many accounts should we bring back, at most? */
|
||||
private final int accountsPerPage;
|
||||
|
||||
/** What page are we on? (1-origin) */
|
||||
private final int pageIndex;
|
||||
|
||||
/** How are they sorted? */
|
||||
private final UserAccountsOrdering orderBy;
|
||||
|
||||
/** What role are we filtering by, if any? */
|
||||
private final String roleFilterUri;
|
||||
|
||||
/** What term are we searching on, if any? */
|
||||
private final String searchTerm;
|
||||
|
||||
public UserAccountsSelectionCriteria(int accountsPerPage, int pageIndex,
|
||||
UserAccountsOrdering orderBy, String roleFilterUri,
|
||||
String searchTerm) {
|
||||
if (accountsPerPage <= 0) {
|
||||
throw new IllegalArgumentException("accountsPerPage "
|
||||
+ "must be a positive integer, not " + accountsPerPage);
|
||||
}
|
||||
this.accountsPerPage = accountsPerPage;
|
||||
|
||||
if (pageIndex <= 0) {
|
||||
throw new IllegalArgumentException("pageIndex must be a "
|
||||
+ "non-negative integer, not " + pageIndex);
|
||||
}
|
||||
this.pageIndex = pageIndex;
|
||||
|
||||
this.orderBy = nonNull(orderBy, UserAccountsOrdering.DEFAULT_ORDERING);
|
||||
|
||||
this.roleFilterUri = nonNull(roleFilterUri, "");
|
||||
this.searchTerm = nonNull(searchTerm, "");
|
||||
}
|
||||
|
||||
public int getAccountsPerPage() {
|
||||
return accountsPerPage;
|
||||
}
|
||||
|
||||
public int getPageIndex() {
|
||||
return pageIndex;
|
||||
}
|
||||
|
||||
public UserAccountsOrdering getOrderBy() {
|
||||
return orderBy;
|
||||
}
|
||||
|
||||
public String getRoleFilterUri() {
|
||||
return roleFilterUri;
|
||||
}
|
||||
|
||||
public String getSearchTerm() {
|
||||
return searchTerm;
|
||||
}
|
||||
|
||||
private <T> T nonNull(T t, T nullValue) {
|
||||
return (t == null) ? nullValue : t;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,353 @@
|
|||
/* $This file is distributed under the terms of the license in /doc/license.txt$ */
|
||||
|
||||
package edu.cornell.mannlib.vitro.webapp.controller.freemarker.accountmanagement;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collection;
|
||||
import java.util.Collections;
|
||||
import java.util.HashSet;
|
||||
import java.util.List;
|
||||
import java.util.Set;
|
||||
|
||||
import org.apache.commons.logging.Log;
|
||||
import org.apache.commons.logging.LogFactory;
|
||||
|
||||
import com.hp.hpl.jena.ontology.OntModel;
|
||||
import com.hp.hpl.jena.query.Query;
|
||||
import com.hp.hpl.jena.query.QueryExecution;
|
||||
import com.hp.hpl.jena.query.QueryExecutionFactory;
|
||||
import com.hp.hpl.jena.query.QueryFactory;
|
||||
import com.hp.hpl.jena.query.QuerySolution;
|
||||
import com.hp.hpl.jena.query.ResultSet;
|
||||
import com.hp.hpl.jena.query.Syntax;
|
||||
import com.hp.hpl.jena.rdf.model.Literal;
|
||||
import com.hp.hpl.jena.rdf.model.Resource;
|
||||
|
||||
import edu.cornell.mannlib.vitro.webapp.beans.UserAccount;
|
||||
import edu.cornell.mannlib.vitro.webapp.beans.UserAccount.Status;
|
||||
import edu.cornell.mannlib.vitro.webapp.controller.freemarker.accountmanagement.UserAccountsOrdering.Field;
|
||||
|
||||
/**
|
||||
* Pull some UserAccounts from the model, based on a set of criteria.
|
||||
*/
|
||||
public class UserAccountsSelector {
|
||||
private static final Log log = LogFactory
|
||||
.getLog(UserAccountsSelector.class);
|
||||
|
||||
private static final String PREFIX_LINES = ""
|
||||
+ "PREFIX rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> \n"
|
||||
+ "PREFIX auth: <http://vitro.mannlib.cornell.edu/ns/vitro/authorization#> \n";
|
||||
|
||||
private static final String ALL_VARIABLES = "?uri ?email ?firstName ?lastName ?pwd ?expire ?count ?status";
|
||||
|
||||
private static final String COUNT_VARIABLE = "?uri";
|
||||
|
||||
private static final String MAIN_QUERY_TEMPLATE = "" //
|
||||
+ "%prefixes% \n" //
|
||||
+ "SELECT DISTINCT %variables% \n" //
|
||||
+ "WHERE {\n" //
|
||||
+ " %requiredClauses% \n" //
|
||||
+ " %optionalClauses% \n" //
|
||||
+ " %filterClauses% \n" //
|
||||
+ "} \n" //
|
||||
+ "ORDER BY %ordering% \n" //
|
||||
+ "LIMIT %limit% \n" //
|
||||
+ "OFFSET %offset% \n";
|
||||
|
||||
private static final String COUNT_QUERY_TEMPLATE = "" //
|
||||
+ "%prefixes% \n" //
|
||||
+ "SELECT count(DISTINCT %countVariable%) \n" //
|
||||
+ "WHERE {\n" //
|
||||
+ " %requiredClauses% \n" //
|
||||
+ " %filterClauses% \n" //
|
||||
+ "} \n";
|
||||
|
||||
private static final String PERMISSIONS_QUERY_TEMPLATE = "" //
|
||||
+ "%prefixes% \n" //
|
||||
+ "SELECT ?ps \n" //
|
||||
+ "WHERE {\n" //
|
||||
+ " <%uri%> auth:hasPermissionSet ?ps \n" //
|
||||
+ "} \n";
|
||||
|
||||
private static final Syntax SYNTAX = Syntax.syntaxARQ;
|
||||
|
||||
/**
|
||||
* Convenience method.
|
||||
*/
|
||||
public static UserAccountsSelection select(OntModel userAccountsModel,
|
||||
UserAccountsSelectionCriteria criteria) {
|
||||
return new UserAccountsSelector(userAccountsModel, criteria).select();
|
||||
}
|
||||
|
||||
private final OntModel model;
|
||||
private final UserAccountsSelectionCriteria criteria;
|
||||
|
||||
public UserAccountsSelector(OntModel userAccountsModel,
|
||||
UserAccountsSelectionCriteria criteria) {
|
||||
if (userAccountsModel == null) {
|
||||
throw new NullPointerException("userAccountsModel may not be null.");
|
||||
}
|
||||
this.model = userAccountsModel;
|
||||
|
||||
if (criteria == null) {
|
||||
throw new NullPointerException("criteria may not be null.");
|
||||
}
|
||||
this.criteria = criteria;
|
||||
}
|
||||
|
||||
public UserAccountsSelection select() {
|
||||
List<UserAccount> accounts = queryForAccounts();
|
||||
int resultCount = queryForCount();
|
||||
queryToPopulatePermissionSets(accounts);
|
||||
return new UserAccountsSelection(criteria, accounts, resultCount);
|
||||
}
|
||||
|
||||
private List<UserAccount> queryForAccounts() {
|
||||
String qString = MAIN_QUERY_TEMPLATE
|
||||
.replace("%prefixes%", PREFIX_LINES)
|
||||
.replace("%variables%", ALL_VARIABLES)
|
||||
.replace("%requiredClauses%", requiredClauses())
|
||||
.replace("%optionalClauses%", optionalClauses())
|
||||
.replace("%filterClauses%", filterClauses())
|
||||
.replace("%ordering%", ordering()).replace("%limit%", limit())
|
||||
.replace("%offset%", offset());
|
||||
log.debug("main query: " + qString);
|
||||
|
||||
List<UserAccount> accounts = executeQuery(qString,
|
||||
new MainQueryParser());
|
||||
log.debug("query returns: " + accounts);
|
||||
return accounts;
|
||||
}
|
||||
|
||||
private int queryForCount() {
|
||||
String qString = COUNT_QUERY_TEMPLATE
|
||||
.replace("%prefixes%", PREFIX_LINES)
|
||||
.replace("%countVariable%", COUNT_VARIABLE)
|
||||
.replace("%requiredClauses%", requiredClauses())
|
||||
.replace("%filterClauses%", filterClauses());
|
||||
log.debug("count query: " + qString);
|
||||
|
||||
int count = executeQuery(qString, new CountQueryParser());
|
||||
log.debug("result count: " + count);
|
||||
return count;
|
||||
}
|
||||
|
||||
private void queryToPopulatePermissionSets(List<UserAccount> accounts) {
|
||||
for (UserAccount account : accounts) {
|
||||
String uri = account.getUri();
|
||||
String qString = PERMISSIONS_QUERY_TEMPLATE.replace("%prefixes%",
|
||||
PREFIX_LINES).replace("%uri%", uri);
|
||||
log.debug("permissions query: " + qString);
|
||||
|
||||
Collection<String> permissions = executeQuery(qString,
|
||||
new PermissionsQueryParser());
|
||||
log.debug("permissions for '" + uri + "': " + permissions);
|
||||
account.setPermissionSetUris(permissions);
|
||||
}
|
||||
}
|
||||
|
||||
/** If the result doesn't have URI and Email Address, we don't want it. */
|
||||
private String requiredClauses() {
|
||||
return "?uri a auth:UserAccount ; \n"
|
||||
+ " auth:emailAddress ?email .";
|
||||
}
|
||||
|
||||
/** If any of these fields are missing, show the result anyway. */
|
||||
private String optionalClauses() {
|
||||
return "OPTIONAL { ?uri auth:firstName ?firstName } \n"
|
||||
+ " OPTIONAL { ?uri auth:lastName ?lastName } \n"
|
||||
+ " OPTIONAL { ?uri auth:md5password ?pwd } \n"
|
||||
+ " OPTIONAL { ?uri auth:passwordChangeExpires ?expire } \n"
|
||||
+ " OPTIONAL { ?uri auth:loginCount ?count } \n"
|
||||
+ " OPTIONAL { ?uri auth:status ?status }";
|
||||
}
|
||||
|
||||
private String filterClauses() {
|
||||
log.warn("UserAccountsSelector.filterClauses() not implemented.");
|
||||
return "";
|
||||
}
|
||||
|
||||
/** Sort as desired, and within ties, sort by EMail address. */
|
||||
private String ordering() {
|
||||
UserAccountsOrdering orderBy = criteria.getOrderBy();
|
||||
String keyword = orderBy.getDirection().keyword;
|
||||
String variable = orderBy.getField().variableName;
|
||||
if (orderBy.getField() == Field.EMAIL) {
|
||||
return keyword + "(?" + variable + ")";
|
||||
} else {
|
||||
return keyword + "(?" + variable + ") ?" + Field.EMAIL.variableName;
|
||||
}
|
||||
}
|
||||
|
||||
private String limit() {
|
||||
return String.valueOf(criteria.getAccountsPerPage());
|
||||
}
|
||||
|
||||
private String offset() {
|
||||
int offset = criteria.getAccountsPerPage()
|
||||
* (criteria.getPageIndex() - 1);
|
||||
return String.valueOf(offset);
|
||||
}
|
||||
|
||||
private <T> T executeQuery(String queryStr, QueryParser<T> parser) {
|
||||
QueryExecution qe = null;
|
||||
try {
|
||||
Query query = QueryFactory.create(queryStr, SYNTAX);
|
||||
qe = QueryExecutionFactory.create(query, model);
|
||||
return parser.parseResults(queryStr, qe.execSelect());
|
||||
} catch (Exception e) {
|
||||
log.error("Failed to execute the query: " + queryStr, e);
|
||||
return parser.defaultValue();
|
||||
} finally {
|
||||
if (qe != null) {
|
||||
qe.close();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static abstract class QueryParser<T> {
|
||||
abstract T parseResults(String queryStr, ResultSet results);
|
||||
|
||||
abstract T defaultValue();
|
||||
|
||||
protected String ifLiteralPresent(QuerySolution solution,
|
||||
String variableName, String defaultValue) {
|
||||
Literal literal = solution.getLiteral(variableName);
|
||||
if (literal == null) {
|
||||
return defaultValue;
|
||||
} else {
|
||||
return literal.getString();
|
||||
}
|
||||
}
|
||||
|
||||
protected long ifLongPresent(QuerySolution solution,
|
||||
String variableName, long defaultValue) {
|
||||
Literal literal = solution.getLiteral(variableName);
|
||||
if (literal == null) {
|
||||
return defaultValue;
|
||||
} else {
|
||||
return literal.getLong();
|
||||
}
|
||||
}
|
||||
|
||||
protected int ifIntPresent(QuerySolution solution, String variableName,
|
||||
int defaultValue) {
|
||||
Literal literal = solution.getLiteral(variableName);
|
||||
if (literal == null) {
|
||||
return defaultValue;
|
||||
} else {
|
||||
return literal.getInt();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
private static class MainQueryParser extends QueryParser<List<UserAccount>> {
|
||||
@Override
|
||||
public List<UserAccount> defaultValue() {
|
||||
return Collections.emptyList();
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<UserAccount> parseResults(String queryStr, ResultSet results) {
|
||||
List<UserAccount> accounts = new ArrayList<UserAccount>();
|
||||
while (results.hasNext()) {
|
||||
try {
|
||||
QuerySolution solution = results.next();
|
||||
UserAccount user = parseSolution(solution);
|
||||
accounts.add(user);
|
||||
} catch (Exception e) {
|
||||
log.warn("Failed to parse the query result: " + queryStr, e);
|
||||
}
|
||||
}
|
||||
return accounts;
|
||||
}
|
||||
|
||||
private UserAccount parseSolution(QuerySolution solution) {
|
||||
UserAccount user = new UserAccount();
|
||||
user.setUri(getUriFromSolution(solution));
|
||||
user.setEmailAddress(solution.getLiteral("email").getString());
|
||||
user.setFirstName(ifLiteralPresent(solution, "firstName", ""));
|
||||
user.setLastName(ifLiteralPresent(solution, "lastName", ""));
|
||||
user.setMd5password(ifLiteralPresent(solution, "pwd", ""));
|
||||
user.setPasswordChangeExpires(ifLongPresent(solution, "expire", 0L));
|
||||
user.setLoginCount(ifIntPresent(solution, "count", 0));
|
||||
user.setStatus(parseStatus(solution, "status", Status.INACTIVE));
|
||||
return user;
|
||||
}
|
||||
|
||||
private Status parseStatus(QuerySolution solution, String variableName,
|
||||
Status defaultValue) {
|
||||
Literal literal = solution.getLiteral(variableName);
|
||||
if (literal == null) {
|
||||
return defaultValue;
|
||||
} else {
|
||||
String string = literal.getString();
|
||||
try {
|
||||
return Status.valueOf(string);
|
||||
} catch (Exception e) {
|
||||
String uri = getUriFromSolution(solution);
|
||||
log.warn("Failed to parse the status value for '" + uri
|
||||
+ "': '" + string + "'");
|
||||
return defaultValue;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private String getUriFromSolution(QuerySolution solution) {
|
||||
return solution.getResource("uri").getURI();
|
||||
}
|
||||
}
|
||||
|
||||
private static class CountQueryParser extends QueryParser<Integer> {
|
||||
@Override
|
||||
public Integer defaultValue() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Integer parseResults(String queryStr, ResultSet results) {
|
||||
int count = 0;
|
||||
|
||||
if (!results.hasNext()) {
|
||||
log.warn("count query returned no results.");
|
||||
}
|
||||
try {
|
||||
QuerySolution solution = results.next();
|
||||
count = ifIntPresent(solution, ".1", 0);
|
||||
} catch (Exception e) {
|
||||
log.warn("Failed to parse the query result" + queryStr, e);
|
||||
}
|
||||
|
||||
return count;
|
||||
}
|
||||
}
|
||||
|
||||
private static class PermissionsQueryParser extends
|
||||
QueryParser<Set<String>> {
|
||||
@Override
|
||||
Set<String> defaultValue() {
|
||||
return Collections.emptySet();
|
||||
}
|
||||
|
||||
@Override
|
||||
Set<String> parseResults(String queryStr, ResultSet results) {
|
||||
Set<String> permissions = new HashSet<String>();
|
||||
|
||||
while (results.hasNext()) {
|
||||
try {
|
||||
QuerySolution solution = results.next();
|
||||
Resource r = solution.getResource("ps");
|
||||
if (r != null) {
|
||||
permissions.add(r.getURI());
|
||||
}
|
||||
} catch (Exception e) {
|
||||
log.warn("Failed to parse the query result: " + queryStr, e);
|
||||
}
|
||||
}
|
||||
|
||||
return permissions;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue