mirror of
https://github.com/google/nomulus.git
synced 2025-05-13 16:07:15 +02:00
Move SendEmailUtils to the /ui/server directory
SendEmailUtils is a general utility of the web console, and not specifically "only" to the Registrar console. ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=226187094
This commit is contained in:
parent
56b61ad5a2
commit
51f22a15ed
8 changed files with 78 additions and 32 deletions
|
@ -509,7 +509,7 @@ public final class RegistryConfig {
|
||||||
/**
|
/**
|
||||||
* The email address that outgoing emails from the app are sent from.
|
* The email address that outgoing emails from the app are sent from.
|
||||||
*
|
*
|
||||||
* @see google.registry.ui.server.registrar.SendEmailUtils
|
* @see google.registry.ui.server.SendEmailUtils
|
||||||
*/
|
*/
|
||||||
@Provides
|
@Provides
|
||||||
@Config("gSuiteOutgoingEmailAddress")
|
@Config("gSuiteOutgoingEmailAddress")
|
||||||
|
@ -520,10 +520,10 @@ public final class RegistryConfig {
|
||||||
/**
|
/**
|
||||||
* The display name that is used on outgoing emails sent by Nomulus.
|
* The display name that is used on outgoing emails sent by Nomulus.
|
||||||
*
|
*
|
||||||
* @see google.registry.ui.server.registrar.SendEmailUtils
|
* @see google.registry.ui.server.SendEmailUtils
|
||||||
*/
|
*/
|
||||||
@Provides
|
@Provides
|
||||||
@Config("gSuiteOutoingEmailDisplayName")
|
@Config("gSuiteOutgoingEmailDisplayName")
|
||||||
public static String provideGSuiteOutgoingEmailDisplayName(RegistryConfigSettings config) {
|
public static String provideGSuiteOutgoingEmailDisplayName(RegistryConfigSettings config) {
|
||||||
return config.gSuite.outgoingEmailDisplayName;
|
return config.gSuite.outgoingEmailDisplayName;
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,12 +13,16 @@ java_library(
|
||||||
"//java/google/registry/ui/css:registrar_dbg.css.js",
|
"//java/google/registry/ui/css:registrar_dbg.css.js",
|
||||||
],
|
],
|
||||||
deps = [
|
deps = [
|
||||||
|
"//java/google/registry/config",
|
||||||
"//java/google/registry/model",
|
"//java/google/registry/model",
|
||||||
"//java/google/registry/ui",
|
"//java/google/registry/ui",
|
||||||
"//java/google/registry/ui/forms",
|
"//java/google/registry/ui/forms",
|
||||||
"//java/google/registry/util",
|
"//java/google/registry/util",
|
||||||
"@com_google_appengine_api_1_0_sdk",
|
"@com_google_appengine_api_1_0_sdk",
|
||||||
"@com_google_code_findbugs_jsr305",
|
"@com_google_code_findbugs_jsr305",
|
||||||
|
"@com_google_dagger",
|
||||||
|
"@com_google_flogger",
|
||||||
|
"@com_google_flogger_system_backend",
|
||||||
"@com_google_guava",
|
"@com_google_guava",
|
||||||
"@com_google_re2j",
|
"@com_google_re2j",
|
||||||
"@io_bazel_rules_closure//closure/templates",
|
"@io_bazel_rules_closure//closure/templates",
|
||||||
|
|
|
@ -12,13 +12,13 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
package google.registry.ui.server.registrar;
|
package google.registry.ui.server;
|
||||||
|
|
||||||
import static com.google.common.collect.ImmutableList.toImmutableList;
|
import static com.google.common.collect.ImmutableList.toImmutableList;
|
||||||
import static com.google.common.collect.Iterables.toArray;
|
import static com.google.common.collect.Iterables.toArray;
|
||||||
|
|
||||||
import com.google.common.base.Joiner;
|
import com.google.common.base.Joiner;
|
||||||
import com.google.common.collect.Streams;
|
import com.google.common.collect.ImmutableList;
|
||||||
import com.google.common.flogger.FluentLogger;
|
import com.google.common.flogger.FluentLogger;
|
||||||
import google.registry.config.RegistryConfig.Config;
|
import google.registry.config.RegistryConfig.Config;
|
||||||
import google.registry.util.SendEmailService;
|
import google.registry.util.SendEmailService;
|
||||||
|
@ -37,30 +37,40 @@ public class SendEmailUtils {
|
||||||
private static final FluentLogger logger = FluentLogger.forEnclosingClass();
|
private static final FluentLogger logger = FluentLogger.forEnclosingClass();
|
||||||
|
|
||||||
private final String gSuiteOutgoingEmailAddress;
|
private final String gSuiteOutgoingEmailAddress;
|
||||||
private final String gSuiteOutoingEmailDisplayName;
|
private final String gSuiteOutgoingEmailDisplayName;
|
||||||
private final SendEmailService emailService;
|
private final SendEmailService emailService;
|
||||||
|
private final ImmutableList<String> registrarChangesNotificationEmailAddresses;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public SendEmailUtils(
|
public SendEmailUtils(
|
||||||
@Config("gSuiteOutgoingEmailAddress") String gSuiteOutgoingEmailAddress,
|
@Config("gSuiteOutgoingEmailAddress") String gSuiteOutgoingEmailAddress,
|
||||||
@Config("gSuiteOutoingEmailDisplayName") String gSuiteOutoingEmailDisplayName,
|
@Config("gSuiteOutgoingEmailDisplayName") String gSuiteOutgoingEmailDisplayName,
|
||||||
|
@Config("registrarChangesNotificationEmailAddresses")
|
||||||
|
ImmutableList<String> registrarChangesNotificationEmailAddresses,
|
||||||
SendEmailService emailService) {
|
SendEmailService emailService) {
|
||||||
this.gSuiteOutgoingEmailAddress = gSuiteOutgoingEmailAddress;
|
this.gSuiteOutgoingEmailAddress = gSuiteOutgoingEmailAddress;
|
||||||
this.gSuiteOutoingEmailDisplayName = gSuiteOutoingEmailDisplayName;
|
this.gSuiteOutgoingEmailDisplayName = gSuiteOutgoingEmailDisplayName;
|
||||||
this.emailService = emailService;
|
this.emailService = emailService;
|
||||||
|
this.registrarChangesNotificationEmailAddresses = registrarChangesNotificationEmailAddresses;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends an email from Nomulus to the specified recipient(s). Returns true iff sending was
|
* Sends an email from Nomulus to the registrarChangesNotificationEmailAddresses. Returns true iff
|
||||||
* successful.
|
* sending to at least 1 address was successful.
|
||||||
|
*
|
||||||
|
* <p>This means that if there are no recepients ({@link #hasRecepients} returns false), this will
|
||||||
|
* return false even thought no error happened.
|
||||||
|
*
|
||||||
|
* <p>This also means that if there are multiple recepients, it will return true even if some (but
|
||||||
|
* not all) of the recepients had an error.
|
||||||
*/
|
*/
|
||||||
public boolean sendEmail(Iterable<String> addresses, final String subject, String body) {
|
public boolean sendEmail(final String subject, String body) {
|
||||||
try {
|
try {
|
||||||
Message msg = emailService.createMessage();
|
Message msg = emailService.createMessage();
|
||||||
msg.setFrom(
|
msg.setFrom(
|
||||||
new InternetAddress(gSuiteOutgoingEmailAddress, gSuiteOutoingEmailDisplayName));
|
new InternetAddress(gSuiteOutgoingEmailAddress, gSuiteOutgoingEmailDisplayName));
|
||||||
List<InternetAddress> emails =
|
List<InternetAddress> emails =
|
||||||
Streams.stream(addresses)
|
registrarChangesNotificationEmailAddresses.stream()
|
||||||
.map(
|
.map(
|
||||||
emailAddress -> {
|
emailAddress -> {
|
||||||
try {
|
try {
|
||||||
|
@ -85,9 +95,17 @@ public class SendEmailUtils {
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
logger.atSevere().withCause(t).log(
|
logger.atSevere().withCause(t).log(
|
||||||
"Could not email to addresses %s with subject '%s'.",
|
"Could not email to addresses %s with subject '%s'.",
|
||||||
Joiner.on(", ").join(addresses), subject);
|
Joiner.on(", ").join(registrarChangesNotificationEmailAddresses), subject);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns whether there are any recepients set up. {@link #sendEmail} will always return false if
|
||||||
|
* there are no recepients.
|
||||||
|
*/
|
||||||
|
public boolean hasRecepients() {
|
||||||
|
return !registrarChangesNotificationEmailAddresses.isEmpty();
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -33,7 +33,6 @@ import com.google.common.collect.Multimap;
|
||||||
import com.google.common.collect.Sets;
|
import com.google.common.collect.Sets;
|
||||||
import com.google.common.collect.Streams;
|
import com.google.common.collect.Streams;
|
||||||
import com.google.common.flogger.FluentLogger;
|
import com.google.common.flogger.FluentLogger;
|
||||||
import google.registry.config.RegistryConfig.Config;
|
|
||||||
import google.registry.config.RegistryEnvironment;
|
import google.registry.config.RegistryEnvironment;
|
||||||
import google.registry.model.registrar.Registrar;
|
import google.registry.model.registrar.Registrar;
|
||||||
import google.registry.model.registrar.RegistrarContact;
|
import google.registry.model.registrar.RegistrarContact;
|
||||||
|
@ -51,6 +50,7 @@ import google.registry.security.JsonResponseHelper;
|
||||||
import google.registry.ui.forms.FormException;
|
import google.registry.ui.forms.FormException;
|
||||||
import google.registry.ui.forms.FormFieldException;
|
import google.registry.ui.forms.FormFieldException;
|
||||||
import google.registry.ui.server.RegistrarFormFields;
|
import google.registry.ui.server.RegistrarFormFields;
|
||||||
|
import google.registry.ui.server.SendEmailUtils;
|
||||||
import google.registry.util.AppEngineServiceUtils;
|
import google.registry.util.AppEngineServiceUtils;
|
||||||
import google.registry.util.CollectionUtils;
|
import google.registry.util.CollectionUtils;
|
||||||
import google.registry.util.DiffUtils;
|
import google.registry.util.DiffUtils;
|
||||||
|
@ -92,8 +92,6 @@ public class RegistrarSettingsAction implements Runnable, JsonActionRunner.JsonA
|
||||||
@Inject AuthResult authResult;
|
@Inject AuthResult authResult;
|
||||||
@Inject RegistryEnvironment registryEnvironment;
|
@Inject RegistryEnvironment registryEnvironment;
|
||||||
|
|
||||||
@Inject @Config("registrarChangesNotificationEmailAddresses") ImmutableList<String>
|
|
||||||
registrarChangesNotificationEmailAddresses;
|
|
||||||
@Inject RegistrarSettingsAction() {}
|
@Inject RegistrarSettingsAction() {}
|
||||||
|
|
||||||
private static final Predicate<RegistrarContact> HAS_PHONE =
|
private static final Predicate<RegistrarContact> HAS_PHONE =
|
||||||
|
@ -478,7 +476,7 @@ public class RegistrarSettingsAction implements Runnable, JsonActionRunner.JsonA
|
||||||
ImmutableSet<RegistrarContact> existingContacts,
|
ImmutableSet<RegistrarContact> existingContacts,
|
||||||
Registrar updatedRegistrar,
|
Registrar updatedRegistrar,
|
||||||
ImmutableSet<RegistrarContact> updatedContacts) {
|
ImmutableSet<RegistrarContact> updatedContacts) {
|
||||||
if (registrarChangesNotificationEmailAddresses.isEmpty()) {
|
if (!sendEmailUtils.hasRecepients()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -495,7 +493,6 @@ public class RegistrarSettingsAction implements Runnable, JsonActionRunner.JsonA
|
||||||
enqueueRegistrarSheetSync(appEngineServiceUtils.getCurrentVersionHostname("backend"));
|
enqueueRegistrarSheetSync(appEngineServiceUtils.getCurrentVersionHostname("backend"));
|
||||||
String environment = Ascii.toLowerCase(String.valueOf(registryEnvironment));
|
String environment = Ascii.toLowerCase(String.valueOf(registryEnvironment));
|
||||||
sendEmailUtils.sendEmail(
|
sendEmailUtils.sendEmail(
|
||||||
registrarChangesNotificationEmailAddresses,
|
|
||||||
String.format(
|
String.format(
|
||||||
"Registrar %s (%s) updated in %s",
|
"Registrar %s (%s) updated in %s",
|
||||||
existingRegistrar.getRegistrarName(),
|
existingRegistrar.getRegistrarName(),
|
||||||
|
|
|
@ -13,12 +13,15 @@ java_library(
|
||||||
deps = [
|
deps = [
|
||||||
"//java/google/registry/ui/forms",
|
"//java/google/registry/ui/forms",
|
||||||
"//java/google/registry/ui/server",
|
"//java/google/registry/ui/server",
|
||||||
|
"//java/google/registry/util",
|
||||||
"//javatests/google/registry/testing",
|
"//javatests/google/registry/testing",
|
||||||
|
"@com_google_appengine_api_1_0_sdk",
|
||||||
"@com_google_guava",
|
"@com_google_guava",
|
||||||
"@com_google_truth",
|
"@com_google_truth",
|
||||||
"@com_google_truth_extensions_truth_java8_extension",
|
"@com_google_truth_extensions_truth_java8_extension",
|
||||||
"@junit",
|
"@junit",
|
||||||
"@org_hamcrest_library",
|
"@org_hamcrest_library",
|
||||||
|
"@org_mockito_all",
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -12,11 +12,9 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
package google.registry.ui.server.registrar;
|
package google.registry.ui.server;
|
||||||
|
|
||||||
import static com.google.common.truth.Truth.assertThat;
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
import static google.registry.config.RegistryConfig.getGSuiteOutgoingEmailAddress;
|
|
||||||
import static google.registry.config.RegistryConfig.getGSuiteOutgoingEmailDisplayName;
|
|
||||||
import static org.mockito.Matchers.any;
|
import static org.mockito.Matchers.any;
|
||||||
import static org.mockito.Mockito.doThrow;
|
import static org.mockito.Mockito.doThrow;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
|
@ -51,16 +49,23 @@ public class SendEmailUtilsTest {
|
||||||
public void init() {
|
public void init() {
|
||||||
message = new MimeMessage(Session.getDefaultInstance(new Properties(), null));
|
message = new MimeMessage(Session.getDefaultInstance(new Properties(), null));
|
||||||
when(emailService.createMessage()).thenReturn(message);
|
when(emailService.createMessage()).thenReturn(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setRecepients(ImmutableList<String> recepients) {
|
||||||
sendEmailUtils =
|
sendEmailUtils =
|
||||||
new SendEmailUtils(
|
new SendEmailUtils(
|
||||||
getGSuiteOutgoingEmailAddress(), getGSuiteOutgoingEmailDisplayName(), emailService);
|
"outgoing@registry.example",
|
||||||
|
"outgoing display name",
|
||||||
|
recepients,
|
||||||
|
emailService);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testSuccess_sendToOneAddress() throws Exception {
|
public void testSuccess_sendToOneAddress() throws Exception {
|
||||||
|
setRecepients(ImmutableList.of("johnny@fakesite.tld"));
|
||||||
|
assertThat(sendEmailUtils.hasRecepients()).isTrue();
|
||||||
assertThat(
|
assertThat(
|
||||||
sendEmailUtils.sendEmail(
|
sendEmailUtils.sendEmail(
|
||||||
ImmutableList.of("johnny@fakesite.tld"),
|
|
||||||
"Welcome to the Internet",
|
"Welcome to the Internet",
|
||||||
"It is a dark and scary place."))
|
"It is a dark and scary place."))
|
||||||
.isTrue();
|
.isTrue();
|
||||||
|
@ -73,9 +78,10 @@ public class SendEmailUtilsTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testSuccess_sendToMultipleAddresses() throws Exception {
|
public void testSuccess_sendToMultipleAddresses() throws Exception {
|
||||||
|
setRecepients(ImmutableList.of("foo@example.com", "bar@example.com"));
|
||||||
|
assertThat(sendEmailUtils.hasRecepients()).isTrue();
|
||||||
assertThat(
|
assertThat(
|
||||||
sendEmailUtils.sendEmail(
|
sendEmailUtils.sendEmail(
|
||||||
ImmutableList.of("foo@example.com", "bar@example.com"),
|
|
||||||
"Welcome to the Internet",
|
"Welcome to the Internet",
|
||||||
"It is a dark and scary place."))
|
"It is a dark and scary place."))
|
||||||
.isTrue();
|
.isTrue();
|
||||||
|
@ -87,9 +93,10 @@ public class SendEmailUtilsTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testSuccess_ignoresMalformedEmailAddress() throws Exception {
|
public void testSuccess_ignoresMalformedEmailAddress() throws Exception {
|
||||||
|
setRecepients(ImmutableList.of("foo@example.com", "1iñvalidemail"));
|
||||||
|
assertThat(sendEmailUtils.hasRecepients()).isTrue();
|
||||||
assertThat(
|
assertThat(
|
||||||
sendEmailUtils.sendEmail(
|
sendEmailUtils.sendEmail(
|
||||||
ImmutableList.of("foo@example.com", "1iñvalidemail"),
|
|
||||||
"Welcome to the Internet",
|
"Welcome to the Internet",
|
||||||
"It is a dark and scary place."))
|
"It is a dark and scary place."))
|
||||||
.isTrue();
|
.isTrue();
|
||||||
|
@ -99,10 +106,23 @@ public class SendEmailUtilsTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testFailure_onlyGivenMalformedAddress() throws Exception {
|
public void testFailure_noAddresses() throws Exception {
|
||||||
|
setRecepients(ImmutableList.of());
|
||||||
|
assertThat(sendEmailUtils.hasRecepients()).isFalse();
|
||||||
|
assertThat(
|
||||||
|
sendEmailUtils.sendEmail(
|
||||||
|
"Welcome to the Internet",
|
||||||
|
"It is a dark and scary place."))
|
||||||
|
.isFalse();
|
||||||
|
verify(emailService, never()).sendMessage(any(Message.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFailure_onlyGivenMalformedAddress() throws Exception {
|
||||||
|
setRecepients(ImmutableList.of("1iñvalidemail"));
|
||||||
|
assertThat(sendEmailUtils.hasRecepients()).isTrue();
|
||||||
assertThat(
|
assertThat(
|
||||||
sendEmailUtils.sendEmail(
|
sendEmailUtils.sendEmail(
|
||||||
ImmutableList.of("1iñvalidemail"),
|
|
||||||
"Welcome to the Internet",
|
"Welcome to the Internet",
|
||||||
"It is a dark and scary place."))
|
"It is a dark and scary place."))
|
||||||
.isFalse();
|
.isFalse();
|
||||||
|
@ -111,10 +131,11 @@ public class SendEmailUtilsTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testFailure_exceptionThrownDuringSend() throws Exception {
|
public void testFailure_exceptionThrownDuringSend() throws Exception {
|
||||||
|
setRecepients(ImmutableList.of("foo@example.com"));
|
||||||
|
assertThat(sendEmailUtils.hasRecepients()).isTrue();
|
||||||
doThrow(new MessagingException()).when(emailService).sendMessage(any(Message.class));
|
doThrow(new MessagingException()).when(emailService).sendMessage(any(Message.class));
|
||||||
assertThat(
|
assertThat(
|
||||||
sendEmailUtils.sendEmail(
|
sendEmailUtils.sendEmail(
|
||||||
ImmutableList.of("foo@example.com"),
|
|
||||||
"Welcome to the Internet",
|
"Welcome to the Internet",
|
||||||
"It is a dark and scary place."))
|
"It is a dark and scary place."))
|
||||||
.isFalse();
|
.isFalse();
|
|
@ -19,6 +19,7 @@ java_library(
|
||||||
"//java/google/registry/request",
|
"//java/google/registry/request",
|
||||||
"//java/google/registry/request/auth",
|
"//java/google/registry/request/auth",
|
||||||
"//java/google/registry/security",
|
"//java/google/registry/security",
|
||||||
|
"//java/google/registry/ui/server",
|
||||||
"//java/google/registry/ui/server/registrar",
|
"//java/google/registry/ui/server/registrar",
|
||||||
"//java/google/registry/util",
|
"//java/google/registry/util",
|
||||||
"//javatests/google/registry/security",
|
"//javatests/google/registry/security",
|
||||||
|
|
|
@ -41,6 +41,7 @@ import google.registry.testing.AppEngineRule;
|
||||||
import google.registry.testing.FakeClock;
|
import google.registry.testing.FakeClock;
|
||||||
import google.registry.testing.InjectRule;
|
import google.registry.testing.InjectRule;
|
||||||
import google.registry.testing.MockitoJUnitRule;
|
import google.registry.testing.MockitoJUnitRule;
|
||||||
|
import google.registry.ui.server.SendEmailUtils;
|
||||||
import google.registry.util.AppEngineServiceUtils;
|
import google.registry.util.AppEngineServiceUtils;
|
||||||
import google.registry.util.SendEmailService;
|
import google.registry.util.SendEmailService;
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
|
@ -97,11 +98,12 @@ public class RegistrarSettingsActionTestCase {
|
||||||
when(appEngineServiceUtils.getCurrentVersionHostname("backend")).thenReturn("backend.hostname");
|
when(appEngineServiceUtils.getCurrentVersionHostname("backend")).thenReturn("backend.hostname");
|
||||||
action.jsonActionRunner = new JsonActionRunner(
|
action.jsonActionRunner = new JsonActionRunner(
|
||||||
ImmutableMap.of(), new JsonResponse(new ResponseImpl(rsp)));
|
ImmutableMap.of(), new JsonResponse(new ResponseImpl(rsp)));
|
||||||
action.registrarChangesNotificationEmailAddresses = ImmutableList.of(
|
|
||||||
"notification@test.example", "notification2@test.example");
|
|
||||||
action.sendEmailUtils =
|
action.sendEmailUtils =
|
||||||
new SendEmailUtils(
|
new SendEmailUtils(
|
||||||
getGSuiteOutgoingEmailAddress(), getGSuiteOutgoingEmailDisplayName(), emailService);
|
getGSuiteOutgoingEmailAddress(),
|
||||||
|
getGSuiteOutgoingEmailDisplayName(),
|
||||||
|
ImmutableList.of("notification@test.example", "notification2@test.example"),
|
||||||
|
emailService);
|
||||||
action.registryEnvironment = RegistryEnvironment.get();
|
action.registryEnvironment = RegistryEnvironment.get();
|
||||||
action.registrarConsoleMetrics = new RegistrarConsoleMetrics();
|
action.registrarConsoleMetrics = new RegistrarConsoleMetrics();
|
||||||
action.authResult =
|
action.authResult =
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue