14-04-2026
Промежуточный комит версии в которой ну хоть какието тестовые уведомления приходят. Но пока ещё вебпуш не работает
This commit is contained in:
parent
24be1d0c1f
commit
21fbc8ffa0
18
doc/mvp-web-push-notes/README.md
Normal file
18
doc/mvp-web-push-notes/README.md
Normal file
@ -0,0 +1,18 @@
|
||||
# MVP notes: Web Push
|
||||
|
||||
## Временное поведение (сделано для тестового стенда)
|
||||
|
||||
- Клиент отправляет push-подписку на сервер при каждом запуске после авторизации, даже если подписка не изменилась.
|
||||
- Причина: на тестовом сервере/после переустановки БД запись о токене может пропасть, а клиент этого не узнает.
|
||||
|
||||
## Что доработать для production
|
||||
|
||||
- Вернуть режим "отправлять только при изменении подписки" как основной.
|
||||
- Добавить безопасный механизм ресинхронизации:
|
||||
- Вариант 1: периодическая принудительная отправка (например, 1 раз в N дней).
|
||||
- Вариант 2: endpoint на сервере "есть ли подписка", и отправка только при отсутствии/рассинхроне.
|
||||
- В логах разделить обычную отправку и принудительную, чтобы видеть лишний трафик.
|
||||
- Добавить e2e-тесты сценариев:
|
||||
- Переустановка сервера (потеря токена в БД).
|
||||
- Смена браузерной подписки.
|
||||
- Повторный запуск клиента без изменений.
|
||||
@ -8,6 +8,7 @@ import {
|
||||
authorizeSession,
|
||||
isSessionInvalidError,
|
||||
refreshSessions,
|
||||
setSessionAuthorizedHandler,
|
||||
setSessionResetHandler,
|
||||
state,
|
||||
terminateCurrentSession,
|
||||
@ -88,6 +89,8 @@ const screenEl = document.getElementById('app-screen');
|
||||
const toolbarEl = document.getElementById('toolbar-slot');
|
||||
|
||||
let currentCleanup = null;
|
||||
let pingIntervalId = null;
|
||||
let sessionRuntimeStarted = false;
|
||||
|
||||
setClientErrorTransport((payload) => authService.reportClientError(payload));
|
||||
|
||||
@ -273,6 +276,29 @@ async function tryAutoLogin() {
|
||||
}
|
||||
}
|
||||
|
||||
async function ensureSessionRuntimeStarted() {
|
||||
if (!state.session.isAuthorized || sessionRuntimeStarted) return;
|
||||
sessionRuntimeStarted = true;
|
||||
|
||||
await initPwaPush({
|
||||
authService,
|
||||
onLog: (entry) => addAppLogEntry(entry),
|
||||
});
|
||||
|
||||
if (pingIntervalId) {
|
||||
window.clearInterval(pingIntervalId);
|
||||
pingIntervalId = null;
|
||||
}
|
||||
pingIntervalId = window.setInterval(async () => {
|
||||
if (!state.session.isAuthorized) return;
|
||||
try {
|
||||
await authService.ws.request('Ping', { timeMs: Date.now() });
|
||||
} catch {
|
||||
// silent keep-alive
|
||||
}
|
||||
}, 60_000);
|
||||
}
|
||||
|
||||
async function init() {
|
||||
addAppLogEntry({
|
||||
level: 'info',
|
||||
@ -281,9 +307,18 @@ async function init() {
|
||||
});
|
||||
|
||||
setSessionResetHandler(() => {
|
||||
sessionRuntimeStarted = false;
|
||||
if (pingIntervalId) {
|
||||
window.clearInterval(pingIntervalId);
|
||||
pingIntervalId = null;
|
||||
}
|
||||
navigate('start-view');
|
||||
});
|
||||
|
||||
setSessionAuthorizedHandler(() => {
|
||||
void ensureSessionRuntimeStarted();
|
||||
});
|
||||
|
||||
if ('serviceWorker' in navigator) {
|
||||
navigator.serviceWorker.addEventListener('message', (event) => {
|
||||
const data = event?.data || {};
|
||||
@ -347,20 +382,7 @@ async function init() {
|
||||
}
|
||||
});
|
||||
await tryAutoLogin();
|
||||
if (state.session.isAuthorized) {
|
||||
await initPwaPush({
|
||||
authService,
|
||||
onLog: (entry) => addAppLogEntry(entry),
|
||||
});
|
||||
window.setInterval(async () => {
|
||||
if (!state.session.isAuthorized) return;
|
||||
try {
|
||||
await authService.ws.request('Ping', { timeMs: Date.now() });
|
||||
} catch {
|
||||
// silent keep-alive
|
||||
}
|
||||
}, 60_000);
|
||||
}
|
||||
await ensureSessionRuntimeStarted();
|
||||
|
||||
if (!window.location.hash) {
|
||||
navigate(state.session.isAuthorized ? 'profile-view' : 'start-view');
|
||||
|
||||
@ -67,6 +67,7 @@ export function render({ navigate, route }) {
|
||||
|
||||
try {
|
||||
await authService.sendDirectMessage({
|
||||
login: state.session.login,
|
||||
toLogin: chatId,
|
||||
text,
|
||||
storagePwd: state.session.storagePwdInMemory,
|
||||
|
||||
@ -1115,14 +1115,14 @@ export class AuthService {
|
||||
return response.payload || {};
|
||||
}
|
||||
|
||||
async sendDirectMessage({ toLogin, text, storagePwd, targetSessionId = null, messageType = 1 }) {
|
||||
async sendDirectMessage({ login, toLogin, text, storagePwd, targetSessionId = null, messageType = 1 }) {
|
||||
const cleanFromLogin = String(login || '').trim();
|
||||
const cleanToLogin = String(toLogin || '').trim();
|
||||
const cleanText = String(text || '');
|
||||
if (!cleanToLogin || !cleanText) throw new Error('Не передан toLogin/text');
|
||||
if (!cleanFromLogin || !cleanToLogin || !cleanText) throw new Error('Не передан login/toLogin/text');
|
||||
if (!storagePwd) throw new Error('Не передан storagePwd для подписи');
|
||||
if (!this.ws.login) throw new Error('Нет активной авторизованной сессии');
|
||||
|
||||
const secrets = await loadEncryptedUserSecrets(this.ws.login, storagePwd);
|
||||
const secrets = await loadEncryptedUserSecrets(cleanFromLogin, storagePwd);
|
||||
const devicePriv = secrets?.deviceKey;
|
||||
if (!devicePriv) throw new Error('Не найден приватный deviceKey');
|
||||
const privateKey = await importPkcs8Ed25519(devicePriv);
|
||||
@ -1130,7 +1130,7 @@ export class AuthService {
|
||||
const prefix = utf8Bytes('SHiNE_msg');
|
||||
const version = uint8Bytes(1);
|
||||
const toBytes = utf8Bytes(cleanToLogin);
|
||||
const fromBytes = utf8Bytes(this.ws.login);
|
||||
const fromBytes = utf8Bytes(cleanFromLogin);
|
||||
if (toBytes.length < 1 || toBytes.length > 30) throw new Error('toLogin должен быть 1..30 ASCII-символов');
|
||||
if (fromBytes.length < 1 || fromBytes.length > 30) throw new Error('fromLogin должен быть 1..30 ASCII-символов');
|
||||
if (cleanText.length > 3000) throw new Error('Слишком длинное сообщение');
|
||||
|
||||
@ -76,15 +76,20 @@ export async function initPwaPush({ authService, onLog = null }) {
|
||||
|
||||
const serialized = JSON.stringify(sub);
|
||||
const prevSerialized = localStorage.getItem(LS_KEY);
|
||||
if (prevSerialized === serialized) {
|
||||
if (prevSerialized !== serialized) {
|
||||
localStorage.setItem(LS_KEY, serialized);
|
||||
log({
|
||||
level: 'info',
|
||||
source: 'web-push',
|
||||
message: 'Push-подписка не изменилась, отправка на сервер не требуется',
|
||||
message: 'Push-подписка изменилась относительно прошлого запуска',
|
||||
});
|
||||
} else {
|
||||
log({
|
||||
level: 'info',
|
||||
source: 'web-push',
|
||||
message: 'Push-подписка не изменилась, но в MVP включена принудительная отправка на сервер',
|
||||
});
|
||||
return;
|
||||
}
|
||||
localStorage.setItem(LS_KEY, serialized);
|
||||
|
||||
const json = sub.toJSON();
|
||||
const endpoint = json.endpoint || '';
|
||||
|
||||
@ -197,6 +197,7 @@ export const state = createInitialState();
|
||||
|
||||
export const authService = new AuthService(state.entrySettings.shineServer);
|
||||
let onSessionReset = null;
|
||||
let onSessionAuthorized = null;
|
||||
|
||||
export function getChatMessages(chatId) {
|
||||
if (!state.chats[chatId]) {
|
||||
@ -321,7 +322,11 @@ export function clearAuthMessages() {
|
||||
state.authUi.info = '';
|
||||
}
|
||||
|
||||
export function authorizeSession({ login, sessionId, storagePwd }) {
|
||||
export function authorizeSession({
|
||||
login = state.session.login,
|
||||
sessionId = state.session.sessionId,
|
||||
storagePwd = state.session.storagePwdInMemory,
|
||||
} = {}) {
|
||||
state.session.isAuthorized = true;
|
||||
state.session.login = login;
|
||||
state.session.sessionId = sessionId;
|
||||
@ -332,12 +337,19 @@ export function authorizeSession({ login, sessionId, storagePwd }) {
|
||||
sessionId,
|
||||
});
|
||||
state.startHint = '';
|
||||
if (onSessionAuthorized) {
|
||||
onSessionAuthorized();
|
||||
}
|
||||
}
|
||||
|
||||
export function setSessionResetHandler(handler) {
|
||||
onSessionReset = typeof handler === 'function' ? handler : null;
|
||||
}
|
||||
|
||||
export function setSessionAuthorizedHandler(handler) {
|
||||
onSessionAuthorized = typeof handler === 'function' ? handler : null;
|
||||
}
|
||||
|
||||
export function isSessionInvalidError(error) {
|
||||
return INVALID_SESSION_CODES.has(error?.code);
|
||||
}
|
||||
|
||||
@ -27,7 +27,7 @@ import static org.junit.jupiter.api.Assertions.fail;
|
||||
*
|
||||
* ВАЖНО:
|
||||
* - НЕ заполняет БД напрямую.
|
||||
* - Создаёт тестовых пользователей A1..A10 через API AddUser.
|
||||
* - Создаёт тестовых пользователей 1..9 через API AddUser.
|
||||
* - Создаёт сеть дружбы через API AddBlock (CONNECTION_FRIEND).
|
||||
*/
|
||||
public class Seed_TestDataPopulation {
|
||||
@ -45,7 +45,7 @@ public class Seed_TestDataPopulation {
|
||||
try (WsSession ws = WsSession.open()) {
|
||||
UserKeys keys = deriveKeysFromPassword(PASSWORD);
|
||||
|
||||
List<String> users = List.of("A1", "A2", "A3", "A4", "A5", "A6", "A7", "A8", "A9", "A10");
|
||||
List<String> users = List.of("1", "2", "3", "4", "5", "6", "7", "8", "9");
|
||||
|
||||
for (String login : users) {
|
||||
createUserViaApi(ws, r, login, keys, t);
|
||||
@ -70,36 +70,38 @@ public class Seed_TestDataPopulation {
|
||||
}
|
||||
|
||||
// Насыщенная сеть дружбы (взаимно). Целевые контрольные значения:
|
||||
// A1=5 друзей, A2=7 друзей (<8), A3=3 друга.
|
||||
addMutualFriend(senders, states, headerHashes, "A1", "A2", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A1", "A3", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A1", "A4", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A1", "A5", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A1", "A6", t);
|
||||
// 1=5 друзей, 2=7 друзей (<8), 3=3 друга.
|
||||
addMutualFriend(senders, states, headerHashes, "1", "2", t);
|
||||
addMutualFriend(senders, states, headerHashes, "1", "3", t);
|
||||
addMutualFriend(senders, states, headerHashes, "1", "4", t);
|
||||
addMutualFriend(senders, states, headerHashes, "1", "5", t);
|
||||
addMutualFriend(senders, states, headerHashes, "1", "6", t);
|
||||
|
||||
addMutualFriend(senders, states, headerHashes, "A2", "A3", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A2", "A4", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A2", "A5", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A2", "A6", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A2", "A7", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A2", "A8", t);
|
||||
addMutualFriend(senders, states, headerHashes, "2", "3", t);
|
||||
addMutualFriend(senders, states, headerHashes, "2", "4", t);
|
||||
addMutualFriend(senders, states, headerHashes, "2", "5", t);
|
||||
addMutualFriend(senders, states, headerHashes, "2", "6", t);
|
||||
addMutualFriend(senders, states, headerHashes, "2", "7", t);
|
||||
addMutualFriend(senders, states, headerHashes, "2", "8", t);
|
||||
|
||||
addMutualFriend(senders, states, headerHashes, "A3", "A4", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A4", "A5", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A5", "A6", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A5", "A7", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A6", "A8", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A6", "A9", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A7", "A10", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A8", "A9", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A8", "A10", t);
|
||||
addMutualFriend(senders, states, headerHashes, "A9", "A10", t);
|
||||
addMutualFriend(senders, states, headerHashes, "3", "4", t);
|
||||
addMutualFriend(senders, states, headerHashes, "4", "5", t);
|
||||
addMutualFriend(senders, states, headerHashes, "5", "6", t);
|
||||
addMutualFriend(senders, states, headerHashes, "5", "7", t);
|
||||
addMutualFriend(senders, states, headerHashes, "6", "8", t);
|
||||
addMutualFriend(senders, states, headerHashes, "6", "9", t);
|
||||
addMutualFriend(senders, states, headerHashes, "8", "9", t);
|
||||
|
||||
verifyOutFriendsCount(ws, r, "A1", 5, t);
|
||||
verifyOutFriendsCount(ws, r, "A2", 7, t);
|
||||
verifyOutFriendsCount(ws, r, "A3", 3, t);
|
||||
// Контакты: 1/2/3 должны быть друг у друга в контактах (взаимно).
|
||||
addMutualContact(senders, states, headerHashes, "1", "2", t);
|
||||
addMutualContact(senders, states, headerHashes, "1", "3", t);
|
||||
addMutualContact(senders, states, headerHashes, "2", "3", t);
|
||||
|
||||
r.ok("Пользователи A1..A10 созданы через API, дружеские связи созданы через AddBlock");
|
||||
verifyOutFriendsCount(ws, r, "1", 5, t);
|
||||
verifyOutFriendsCount(ws, r, "2", 7, t);
|
||||
verifyOutFriendsCount(ws, r, "3", 3, t);
|
||||
|
||||
r.ok("Пользователи 1..9 созданы через API, дружеские и контактные связи созданы через AddBlock");
|
||||
} catch (Throwable e) {
|
||||
r.fail("Ошибка IT_07: " + e.getMessage());
|
||||
fail("IT_07 failed", e);
|
||||
@ -159,13 +161,30 @@ public class Seed_TestDataPopulation {
|
||||
String targetBch,
|
||||
byte[] targetHeaderHash,
|
||||
Duration timeout) {
|
||||
sendConnection(sender, st, MsgSubType.CONNECTION_FRIEND, targetBch, targetHeaderHash, timeout);
|
||||
}
|
||||
|
||||
private static void sendContactConnection(AddBlockSender sender,
|
||||
ChainState st,
|
||||
String targetBch,
|
||||
byte[] targetHeaderHash,
|
||||
Duration timeout) {
|
||||
sendConnection(sender, st, MsgSubType.CONNECTION_CONTACT, targetBch, targetHeaderHash, timeout);
|
||||
}
|
||||
|
||||
private static void sendConnection(AddBlockSender sender,
|
||||
ChainState st,
|
||||
short relationSubType,
|
||||
String targetBch,
|
||||
byte[] targetHeaderHash,
|
||||
Duration timeout) {
|
||||
ChainState.NextLine ln = st.nextLineByType(ChainState.TYPE_CONNECTION);
|
||||
sender.send(new ConnectionBody(
|
||||
0,
|
||||
ln.prevLineNumber,
|
||||
ln.prevLineHash32,
|
||||
ln.thisLineNumber,
|
||||
MsgSubType.CONNECTION_FRIEND,
|
||||
relationSubType,
|
||||
targetBch,
|
||||
0,
|
||||
targetHeaderHash
|
||||
@ -182,6 +201,16 @@ public class Seed_TestDataPopulation {
|
||||
sendFriendConnection(senders.get(b), states.get(b), bch(a), headerHashes.get(a), t);
|
||||
}
|
||||
|
||||
private static void addMutualContact(Map<String, AddBlockSender> senders,
|
||||
Map<String, ChainState> states,
|
||||
Map<String, byte[]> headerHashes,
|
||||
String a,
|
||||
String b,
|
||||
Duration t) {
|
||||
sendContactConnection(senders.get(a), states.get(a), bch(b), headerHashes.get(b), t);
|
||||
sendContactConnection(senders.get(b), states.get(b), bch(a), headerHashes.get(a), t);
|
||||
}
|
||||
|
||||
private static void verifyOutFriendsCount(WsSession ws, TestResult r, String login, int expectedCount, Duration t) {
|
||||
String resp = ws.call("GetFriendsLists#" + login, JsonBuilders.getFriendsLists(login), t);
|
||||
int st = JsonParsers.status(resp);
|
||||
|
||||
Loading…
Reference in New Issue
Block a user