Enable VAPID
This commit is contained in:
5
pom.xml
5
pom.xml
@@ -66,6 +66,11 @@
|
|||||||
<artifactId>bcprov-jdk15on</artifactId>
|
<artifactId>bcprov-jdk15on</artifactId>
|
||||||
<version>1.64</version>
|
<version>1.64</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.bouncycastle</groupId>
|
||||||
|
<artifactId>bcpkix-jdk15on</artifactId>
|
||||||
|
<version>1.64</version>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.apache.commons</groupId>
|
<groupId>org.apache.commons</groupId>
|
||||||
<artifactId>commons-collections4</artifactId>
|
<artifactId>commons-collections4</artifactId>
|
||||||
|
|||||||
@@ -6,11 +6,12 @@ import org.slf4j.Logger;
|
|||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.stereotype.Controller;
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
import org.springframework.web.bind.annotation.PostMapping;
|
import org.springframework.web.bind.annotation.PostMapping;
|
||||||
import org.springframework.web.bind.annotation.RequestBody;
|
import org.springframework.web.bind.annotation.RequestBody;
|
||||||
|
import org.springframework.web.bind.annotation.RestController;
|
||||||
|
|
||||||
@Controller
|
@RestController
|
||||||
public class SubscriptionController {
|
public class SubscriptionController {
|
||||||
private static final Logger LOGGER = LoggerFactory.getLogger(SubscriptionController.class);
|
private static final Logger LOGGER = LoggerFactory.getLogger(SubscriptionController.class);
|
||||||
|
|
||||||
@@ -23,4 +24,9 @@ public class SubscriptionController {
|
|||||||
|
|
||||||
return this.subscriptionService.subscribe(subscriptionDto).toResponseEntity();
|
return this.subscriptionService.subscribe(subscriptionDto).toResponseEntity();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@GetMapping("subscription/vapidPublicKey")
|
||||||
|
public String getVapidPublicKey() {
|
||||||
|
return this.subscriptionService.getVapidPublicKey();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package de.pushservice.client.dto;
|
package de.pushservice.client.dto;
|
||||||
|
|
||||||
import de.pushservice.client.model.Urgency;
|
import de.pushservice.client.model.Urgency;
|
||||||
|
import org.apache.commons.lang3.builder.ReflectionToStringBuilder;
|
||||||
|
|
||||||
public class MessageDto {
|
public class MessageDto {
|
||||||
private String payload;
|
private String payload;
|
||||||
@@ -36,4 +37,9 @@ public class MessageDto {
|
|||||||
public void setUrgency(Urgency urgency) {
|
public void setUrgency(Urgency urgency) {
|
||||||
this.urgency = urgency;
|
this.urgency = urgency;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return ReflectionToStringBuilder.toString(this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package de.pushservice.client.dto;
|
package de.pushservice.client.dto;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.builder.ReflectionToStringBuilder;
|
||||||
|
|
||||||
public class NotificationRequestDto {
|
public class NotificationRequestDto {
|
||||||
private String scope;
|
private String scope;
|
||||||
private MessageDto messageDto;
|
private MessageDto messageDto;
|
||||||
@@ -24,4 +26,9 @@ public class NotificationRequestDto {
|
|||||||
public void setMessageDto(MessageDto messageDto) {
|
public void setMessageDto(MessageDto messageDto) {
|
||||||
this.messageDto = messageDto;
|
this.messageDto = messageDto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return ReflectionToStringBuilder.toString(this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package de.pushservice.client.dto;
|
package de.pushservice.client.dto;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.builder.ReflectionToStringBuilder;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>
|
* <p>
|
||||||
* Payload for a notification. Gets sent to the third party push service which forwards it to the user agent.
|
* Payload for a notification. Gets sent to the third party push service which forwards it to the user agent.
|
||||||
@@ -60,4 +62,9 @@ public class PayloadDto {
|
|||||||
public void setTitle(String title) {
|
public void setTitle(String title) {
|
||||||
this.title = title;
|
this.title = title;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return ReflectionToStringBuilder.toString(this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package de.pushservice.client.dto;
|
package de.pushservice.client.dto;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.builder.ReflectionToStringBuilder;
|
||||||
|
|
||||||
public class SubscriptionDto {
|
public class SubscriptionDto {
|
||||||
private String auth;
|
private String auth;
|
||||||
private String endpoint;
|
private String endpoint;
|
||||||
@@ -37,4 +39,9 @@ public class SubscriptionDto {
|
|||||||
public void setScope(String scope) {
|
public void setScope(String scope) {
|
||||||
this.scope = scope;
|
this.scope = scope;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return ReflectionToStringBuilder.toString(this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -68,4 +68,11 @@ public class SubscriptionService {
|
|||||||
LOGGER.error(String.format("Could not send notification! %s", responseReason));
|
LOGGER.error(String.format("Could not send notification! %s", responseReason));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getVapidPublicKey() {
|
||||||
|
final ResponseEntity<String> responseEntity = new RestTemplate()
|
||||||
|
.exchange(this.config.getServerUrl() + "vapidPublicKey", HttpMethod.GET, HttpEntity.EMPTY, String.class);
|
||||||
|
|
||||||
|
return responseEntity.getBody();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -45,9 +45,20 @@ function initializeState(registrationScope) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function subscribe(registrationScope) {
|
async function subscribe(registrationScope) {
|
||||||
|
let response = await fetch('subscription/vapidPublicKey');
|
||||||
|
let key = await response.text();
|
||||||
|
options = {};
|
||||||
|
|
||||||
|
if (!key) {
|
||||||
|
options = {userVisibleOnly: true};
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
options = {userVisibleOnly: true, applicationServerKey: base64UrlToUint8Array(key)};
|
||||||
|
}
|
||||||
|
|
||||||
navigator.serviceWorker.ready.then(function (serviceWorkerRegistration) {
|
navigator.serviceWorker.ready.then(function (serviceWorkerRegistration) {
|
||||||
serviceWorkerRegistration.pushManager.subscribe({userVisibleOnly: true}).then(function (subscription) {
|
serviceWorkerRegistration.pushManager.subscribe(options).then(function (subscription) {
|
||||||
return sendSubscriptionToServer(subscription, registrationScope);
|
return sendSubscriptionToServer(subscription, registrationScope);
|
||||||
})
|
})
|
||||||
.catch(function (e) {
|
.catch(function (e) {
|
||||||
@@ -80,4 +91,20 @@ function sendSubscriptionToServer(subscription, registrationScope) {
|
|||||||
scope: registrationScope
|
scope: registrationScope
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function base64UrlToUint8Array(base64UrlData) {
|
||||||
|
const padding = '='.repeat((4 - base64UrlData.length % 4) % 4);
|
||||||
|
const base64 = (base64UrlData + padding)
|
||||||
|
.replace(/\-/g, '+')
|
||||||
|
.replace(/_/g, '/');
|
||||||
|
|
||||||
|
const rawData = atob(base64);
|
||||||
|
const buffer = new Uint8Array(rawData.length);
|
||||||
|
|
||||||
|
for (let i = 0; i < rawData.length; ++i) {
|
||||||
|
buffer[i] = rawData.charCodeAt(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
return buffer;
|
||||||
}
|
}
|
||||||
@@ -45,6 +45,10 @@
|
|||||||
<groupId>org.bouncycastle</groupId>
|
<groupId>org.bouncycastle</groupId>
|
||||||
<artifactId>bcprov-jdk15on</artifactId>
|
<artifactId>bcprov-jdk15on</artifactId>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.bouncycastle</groupId>
|
||||||
|
<artifactId>bcpkix-jdk15on</artifactId>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.flywaydb</groupId>
|
<groupId>org.flywaydb</groupId>
|
||||||
<artifactId>flyway-core</artifactId>
|
<artifactId>flyway-core</artifactId>
|
||||||
|
|||||||
@@ -0,0 +1,39 @@
|
|||||||
|
package de.pushservice.server.config;
|
||||||
|
|
||||||
|
import org.springframework.boot.context.properties.ConfigurationProperties;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
@ConfigurationProperties(prefix = "push-service-server")
|
||||||
|
public class PushServiceServerConfig {
|
||||||
|
public static class Vapid {
|
||||||
|
private String publicKeyPath;
|
||||||
|
private String privateKeyPath;
|
||||||
|
|
||||||
|
public String getPublicKeyPath() {
|
||||||
|
return publicKeyPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPublicKeyPath(String publicKeyPath) {
|
||||||
|
this.publicKeyPath = publicKeyPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getPrivateKeyPath() {
|
||||||
|
return privateKeyPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPrivateKeyPath(String privateKeyPath) {
|
||||||
|
this.privateKeyPath = privateKeyPath;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Vapid vapid;
|
||||||
|
|
||||||
|
public Vapid getVapid() {
|
||||||
|
return vapid;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setVapid(Vapid vapid) {
|
||||||
|
this.vapid = vapid;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -8,6 +8,7 @@ import org.slf4j.Logger;
|
|||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
import org.springframework.web.bind.annotation.PostMapping;
|
import org.springframework.web.bind.annotation.PostMapping;
|
||||||
import org.springframework.web.bind.annotation.RequestBody;
|
import org.springframework.web.bind.annotation.RequestBody;
|
||||||
import org.springframework.web.bind.annotation.RestController;
|
import org.springframework.web.bind.annotation.RestController;
|
||||||
@@ -22,16 +23,29 @@ public class PushServiceController {
|
|||||||
private PushService pushService;
|
private PushService pushService;
|
||||||
|
|
||||||
static {
|
static {
|
||||||
Security.addProvider(new BouncyCastleProvider());
|
if (Security.getProvider(BouncyCastleProvider.PROVIDER_NAME) == null) {
|
||||||
|
Security.addProvider(new BouncyCastleProvider());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@PostMapping("/notify")
|
@PostMapping("/notify")
|
||||||
public ResponseEntity notify(@RequestBody NotificationRequestDto notificationRequestDto) {
|
public ResponseEntity notify(@RequestBody NotificationRequestDto notificationRequestDto) {
|
||||||
|
LOGGER.debug(String.format("notify called with %s", notificationRequestDto));
|
||||||
|
|
||||||
return this.pushService.notify(notificationRequestDto).toResponseEntity();
|
return this.pushService.notify(notificationRequestDto).toResponseEntity();
|
||||||
}
|
}
|
||||||
|
|
||||||
@PostMapping("/subscribe")
|
@PostMapping("/subscribe")
|
||||||
public ResponseEntity subscribe(@RequestBody SubscriptionDto subscriptionDto) {
|
public ResponseEntity subscribe(@RequestBody SubscriptionDto subscriptionDto) {
|
||||||
|
LOGGER.debug(String.format("subscribe called with %s", subscriptionDto));
|
||||||
|
|
||||||
return this.pushService.subscribe(subscriptionDto).toResponseEntity();
|
return this.pushService.subscribe(subscriptionDto).toResponseEntity();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@GetMapping("/vapidPublicKey")
|
||||||
|
public String getVapidPublicKey() {
|
||||||
|
LOGGER.debug("vapidPublicKey called");
|
||||||
|
|
||||||
|
return this.pushService.getVapidPublicKey();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,12 +4,18 @@ import de.pushservice.client.ResponseReason;
|
|||||||
import de.pushservice.client.dto.MessageDto;
|
import de.pushservice.client.dto.MessageDto;
|
||||||
import de.pushservice.client.dto.NotificationRequestDto;
|
import de.pushservice.client.dto.NotificationRequestDto;
|
||||||
import de.pushservice.client.dto.SubscriptionDto;
|
import de.pushservice.client.dto.SubscriptionDto;
|
||||||
|
import de.pushservice.server.config.PushServiceServerConfig;
|
||||||
import de.pushservice.server.dba.SubscriptionRepository;
|
import de.pushservice.server.dba.SubscriptionRepository;
|
||||||
import de.pushservice.server.decorator.SubscriptionDecorator;
|
import de.pushservice.server.decorator.SubscriptionDecorator;
|
||||||
import de.pushservice.server.model.Subscription;
|
import de.pushservice.server.model.Subscription;
|
||||||
import nl.martijndwars.webpush.Notification;
|
import nl.martijndwars.webpush.Notification;
|
||||||
import nl.martijndwars.webpush.Urgency;
|
import nl.martijndwars.webpush.Urgency;
|
||||||
|
import org.apache.commons.collections4.IterableUtils;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
|
import org.bouncycastle.asn1.x509.SubjectPublicKeyInfo;
|
||||||
|
import org.bouncycastle.openssl.PEMKeyPair;
|
||||||
|
import org.bouncycastle.openssl.PEMParser;
|
||||||
|
import org.bouncycastle.openssl.jcajce.JcaPEMKeyConverter;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
@@ -17,7 +23,13 @@ import org.springframework.stereotype.Service;
|
|||||||
import org.springframework.transaction.annotation.Propagation;
|
import org.springframework.transaction.annotation.Propagation;
|
||||||
import org.springframework.transaction.annotation.Transactional;
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileReader;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.security.KeyPair;
|
||||||
import java.time.LocalDateTime;
|
import java.time.LocalDateTime;
|
||||||
|
import java.util.Base64;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
public class PushService {
|
public class PushService {
|
||||||
@@ -29,10 +41,21 @@ public class PushService {
|
|||||||
@Autowired
|
@Autowired
|
||||||
private ExternalPushServiceResponseHandler handler;
|
private ExternalPushServiceResponseHandler handler;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private PushServiceServerConfig config;
|
||||||
|
|
||||||
@Transactional(propagation = Propagation.REQUIRED)
|
@Transactional(propagation = Propagation.REQUIRED)
|
||||||
public ResponseReason subscribe(SubscriptionDto subscriptionDto) {
|
public ResponseReason subscribe(SubscriptionDto subscriptionDto) {
|
||||||
if (this.subscriptionRepository
|
Optional<Subscription> optSubscription = this.subscriptionRepository
|
||||||
.getForScopeAndEndpoint(subscriptionDto.getScope(), subscriptionDto.getEndpoint()).isPresent()) {
|
.getForScopeAndEndpoint(subscriptionDto.getScope(), subscriptionDto.getEndpoint());
|
||||||
|
|
||||||
|
if (optSubscription.isPresent()) {
|
||||||
|
final Subscription subscription = optSubscription.get();
|
||||||
|
|
||||||
|
subscription.setLastSeen(LocalDateTime.now());
|
||||||
|
|
||||||
|
this.subscriptionRepository.save(subscription);
|
||||||
|
|
||||||
return ResponseReason.OK;
|
return ResponseReason.OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -58,9 +81,14 @@ public class PushService {
|
|||||||
|
|
||||||
final Iterable<Subscription> subscriptions = this.subscriptionRepository
|
final Iterable<Subscription> subscriptions = this.subscriptionRepository
|
||||||
.getAllForScope(notificationRequestDto.getScope());
|
.getAllForScope(notificationRequestDto.getScope());
|
||||||
final nl.martijndwars.webpush.PushService extPushService = new nl.martijndwars.webpush.PushService();
|
|
||||||
|
LOGGER.debug(String.format("Found %s subscriptions for scope %s", IterableUtils.size(subscriptions), notificationRequestDto.getScope()));
|
||||||
|
|
||||||
|
final nl.martijndwars.webpush.PushService extPushService = new nl.martijndwars.webpush.PushService(getKeyPair());
|
||||||
|
|
||||||
for (Subscription subscription : subscriptions) {
|
for (Subscription subscription : subscriptions) {
|
||||||
|
LOGGER.debug(String.format("Process endpoint %s", subscription.getEndpoint()));
|
||||||
|
|
||||||
try {
|
try {
|
||||||
SubscriptionDecorator subscriptionDecorator = SubscriptionDecorator.from(subscription);
|
SubscriptionDecorator subscriptionDecorator = SubscriptionDecorator.from(subscription);
|
||||||
Notification notification = Notification.builder()
|
Notification notification = Notification.builder()
|
||||||
@@ -87,4 +115,51 @@ public class PushService {
|
|||||||
private Urgency getExtUrgency(MessageDto messageDto) {
|
private Urgency getExtUrgency(MessageDto messageDto) {
|
||||||
return Urgency.valueOf(messageDto.getUrgency().name());
|
return Urgency.valueOf(messageDto.getUrgency().name());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getVapidPublicKey() {
|
||||||
|
if (StringUtils.isEmpty(this.config.getVapid().getPublicKeyPath())) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final File publicKeyFile = new File(this.config.getVapid().getPublicKeyPath());
|
||||||
|
|
||||||
|
if (!publicKeyFile.exists()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try (FileReader fr = new FileReader(publicKeyFile)) {
|
||||||
|
PEMParser pemParser = new PEMParser(fr);
|
||||||
|
SubjectPublicKeyInfo keyInfo = (SubjectPublicKeyInfo) pemParser.readObject();
|
||||||
|
|
||||||
|
return Base64.getEncoder().encodeToString(keyInfo.getPublicKeyData().getBytes());
|
||||||
|
}
|
||||||
|
catch (IOException ioe) {
|
||||||
|
LOGGER.error(String.format("Could not read key! %s", this.config.getVapid().getPublicKeyPath()), ioe);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private KeyPair getKeyPair() {
|
||||||
|
if (StringUtils.isEmpty(this.config.getVapid().getPrivateKeyPath())) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final File publicKeyFile = new File(this.config.getVapid().getPrivateKeyPath());
|
||||||
|
|
||||||
|
if (!publicKeyFile.exists()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try (FileReader fr = new FileReader(publicKeyFile)) {
|
||||||
|
PEMParser pemParser = new PEMParser(fr);
|
||||||
|
|
||||||
|
return new JcaPEMKeyConverter().getKeyPair((PEMKeyPair) pemParser.readObject());
|
||||||
|
}
|
||||||
|
catch (IOException ioe) {
|
||||||
|
LOGGER.error(String.format("Could not read key! %s", this.config.getVapid().getPrivateKeyPath()), ioe);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
spring.datasource.url=jdbc:postgresql://localhost/push_service_mk
|
spring.datasource.url=jdbc:postgresql://localhost/push_service_mk
|
||||||
spring.datasource.username=push_service_mk
|
spring.datasource.username=push_service_mk
|
||||||
spring.datasource.password=push_service_mk
|
spring.datasource.password=push_service_mk
|
||||||
|
|
||||||
|
push-service-server.vapid.public-key-path=/opt/push-service/mk/vapid_public.pem
|
||||||
|
push-service-server.vapid.private-key-path=/opt/push-service/mk/vapid_private.pem
|
||||||
@@ -18,4 +18,8 @@ logging.file.max-size=50MB
|
|||||||
# because the connection pool cannot shutdown properly
|
# because the connection pool cannot shutdown properly
|
||||||
spring.jmx.enabled=false
|
spring.jmx.enabled=false
|
||||||
|
|
||||||
spring.jpa.hibernate.ddl-auto=validate
|
spring.jpa.hibernate.ddl-auto=validate
|
||||||
|
|
||||||
|
# Follow https://github.com/web-push-libs/webpush-java/wiki/VAPID how to create the keys
|
||||||
|
push-service-server.vapid.public-key-path=/tmp/vapid_public.pem
|
||||||
|
push-service-server.vapid.private-key-path=/tmp/vapid_private.pem
|
||||||
Reference in New Issue
Block a user