-
-
Notifications
You must be signed in to change notification settings - Fork 44
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Improve to manage notifications and proposals #2046
Changes from all commits
dfe9844
0f5d9c9
13f132d
8972c6a
3e83930
8b7de9c
9b77f81
8b8b101
fea1963
be2ceb8
67405c1
8e7d0d4
2e710b3
4c138d3
3be4f34
07f200c
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -848,13 +848,16 @@ sbp('chelonia/defineContract', { | |
const payment = state.payments[data.paymentHash] | ||
|
||
if (loggedIn.identityContractID === payment.data.toMemberID) { | ||
sbp('gi.contracts/group/emitNotificationAfterSyncing', [contractID, innerSigningContractID], 'PAYMENT_RECEIVED', { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
creatorID: innerSigningContractID, | ||
paymentHash: data.paymentHash, | ||
amount: getters.withGroupCurrency(payment.data.amount) | ||
}) | ||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', [contractID, innerSigningContractID], [{ | ||
notificationName: 'PAYMENT_RECEIVED', | ||
notificationData: { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
creatorID: innerSigningContractID, | ||
paymentHash: data.paymentHash, | ||
amount: getters.withGroupCurrency(payment.data.amount) | ||
} | ||
}]) | ||
} | ||
} | ||
} | ||
|
@@ -872,12 +875,15 @@ sbp('chelonia/defineContract', { | |
const { loggedIn } = sbp('state/vuex/state') | ||
|
||
if (data.toMemberID === loggedIn.identityContractID) { | ||
sbp('gi.contracts/group/emitNotificationAfterSyncing', [contractID, innerSigningContractID], 'PAYMENT_THANKYOU_SENT', { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
fromMemberID: innerSigningContractID, | ||
toMemberID: data.toMemberID | ||
}) | ||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', [contractID, innerSigningContractID], [{ | ||
notificationName: 'PAYMENT_THANKYOU_SENT', | ||
notificationData: { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
fromMemberID: innerSigningContractID, | ||
toMemberID: data.toMemberID | ||
} | ||
}]) | ||
} | ||
} | ||
}, | ||
|
@@ -937,12 +943,15 @@ sbp('chelonia/defineContract', { | |
const myProfile = getters.groupProfile(loggedIn.identityContractID) | ||
|
||
if (isActionOlderThanUser(contractID, height, myProfile)) { | ||
sbp('gi.contracts/group/emitNotificationAfterSyncing', [contractID, innerSigningContractID], 'NEW_PROPOSAL', { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
creatorID: innerSigningContractID, | ||
subtype: typeToSubTypeMap[data.proposalType] | ||
}) | ||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', [contractID, innerSigningContractID], [{ | ||
notificationName: 'NEW_PROPOSAL', | ||
notificationData: { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
creatorID: innerSigningContractID, | ||
subtype: typeToSubTypeMap[data.proposalType] | ||
} | ||
}]) | ||
} | ||
} | ||
}, | ||
|
@@ -1024,11 +1033,25 @@ sbp('chelonia/defineContract', { | |
} | ||
}, | ||
'gi.contracts/group/notifyExpiringProposals': { | ||
validate: actionRequireActiveMember(arrayOf(string)), | ||
validate: actionRequireActiveMember(objectOf({ | ||
proposalIds: arrayOf(string) | ||
})), | ||
process ({ data }, { state }) { | ||
for (const proposalId of data) { | ||
for (const proposalId of data.proposalIds) { | ||
Vue.set(state.proposals[proposalId], 'notifiedBeforeExpire', true) | ||
} | ||
}, | ||
sideEffect ({ data, contractID }, { state }) { | ||
const notifications = [] | ||
for (const proposalId of data.proposalIds) { | ||
const proposal = state.proposals[proposalId] | ||
notifications.push({ | ||
notificationName: 'PROPOSAL_EXPIRING', | ||
notificationData: { groupID: contractID, proposal, proposalId } | ||
}) | ||
} | ||
|
||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', contractID, notifications) | ||
} | ||
}, | ||
'gi.contracts/group/removeMember': { | ||
|
@@ -1197,11 +1220,14 @@ sbp('chelonia/defineContract', { | |
const myProfile = profiles[userID] | ||
|
||
if (isActionOlderThanUser(contractID, height, myProfile)) { | ||
sbp('gi.notifications/emit', 'MEMBER_ADDED', { // emit a notification for a member addition. | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
memberID: innerSigningContractID | ||
}) | ||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', [], [{ | ||
notificationName: 'MEMBER_ADDED', | ||
notificationData: { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
memberID: innerSigningContractID | ||
} | ||
}]) | ||
} | ||
}).catch((e) => { | ||
console.error(`Error subscribing to identity contract ${innerSigningContractID} of group member for group ${contractID}`, e) | ||
|
@@ -1687,9 +1713,10 @@ sbp('chelonia/defineContract', { | |
const { loggedIn } = sbp('state/vuex/state') | ||
const { createdDate } = meta | ||
if (isActionOlderThanUser(contractID, height, state.profiles[loggedIn.identityContractID])) { | ||
sbp('gi.contracts/group/emitNotificationAfterSyncing', contractID, 'PROPOSAL_CLOSED', { | ||
createdDate, groupID: contractID, proposal | ||
}) | ||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', contractID, [{ | ||
notificationName: 'PROPOSAL_CLOSED', | ||
notificationData: { createdDate, groupID: contractID, proposal } | ||
}]) | ||
} | ||
}, | ||
'gi.contracts/group/sendMincomeChangedNotification': async function (contractID, meta, data, height, innerSigningContractID) { | ||
|
@@ -1732,13 +1759,16 @@ sbp('chelonia/defineContract', { | |
}) | ||
} | ||
|
||
sbp('gi.contracts/group/emitNotificationAfterSyncing', [contractID, innerSigningContractID], 'MINCOME_CHANGED', { | ||
groupID: contractID, | ||
creatorID: innerSigningContractID, | ||
to: toAmount, | ||
memberType, | ||
increased: mincomeIncreased | ||
}) | ||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', [contractID, innerSigningContractID], [{ | ||
notificationName: 'MINCOME_CHANGED', | ||
notificationData: { | ||
groupID: contractID, | ||
creatorID: innerSigningContractID, | ||
to: toAmount, | ||
memberType, | ||
increased: mincomeIncreased | ||
} | ||
}]) | ||
} | ||
}, | ||
'gi.contracts/group/joinGroupChatrooms': async function (contractID, chatRoomID, memberID) { | ||
|
@@ -1889,12 +1919,10 @@ sbp('chelonia/defineContract', { | |
if (!proposalHash) { | ||
// NOTE: Do not make notification when the member is removed by proposal | ||
const memberRemovedThemselves = memberID === innerSigningContractID | ||
const notificationName = memberRemovedThemselves ? 'MEMBER_LEFT' : 'MEMBER_REMOVED' | ||
sbp('gi.contracts/group/emitNotificationAfterSyncing', memberID, notificationName, { | ||
createdDate: meta.createdDate, | ||
groupID: contractID, | ||
memberID | ||
}) | ||
sbp('gi.contracts/group/emitNotificationsAfterSyncing', memberID, [{ | ||
notificationName: memberRemovedThemselves ? 'MEMBER_LEFT' : 'MEMBER_REMOVED', | ||
notificationData: { createdDate: meta.createdDate, groupID: contractID, memberID } | ||
}]) | ||
} | ||
|
||
Promise.resolve() | ||
|
@@ -1953,13 +1981,15 @@ sbp('chelonia/defineContract', { | |
console.warn(`removeForeignKeys: ${e.name} error thrown:`, e) | ||
}) | ||
}, | ||
'gi.contracts/group/emitNotificationAfterSyncing': async (contractIDs, notificationName, notificationData) => { | ||
'gi.contracts/group/emitNotificationsAfterSyncing': async (contractIDs, notifications) => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Updated function to be able to emit multiple notifications. |
||
const listOfIds = typeof contractIDs === 'string' ? [contractIDs] : contractIDs | ||
for (const id of listOfIds) { | ||
await sbp('chelonia/contract/wait', id) | ||
} | ||
|
||
sbp('gi.notifications/emit', notificationName, notificationData) | ||
notifications.forEach(({ notificationName, notificationData }) => { | ||
sbp('gi.notifications/emit', notificationName, notificationData) | ||
}) | ||
} | ||
} | ||
}) |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -35,6 +35,7 @@ export const MAX_GROUP_MEMBER_COUNT = 150 // Dunbar's number (https://en.wikiped | |
export const STATUS_OPEN = 'open' | ||
export const STATUS_PASSED = 'passed' | ||
export const STATUS_FAILED = 'failed' | ||
export const STATUS_EXPIRING = 'expiring' // Only useful to notify users that the proposals are expiring | ||
export const STATUS_EXPIRED = 'expired' | ||
export const STATUS_CANCELLED = 'cancelled' | ||
|
||
|
@@ -55,7 +56,6 @@ export const CHATROOM_MEMBER_MENTION_SPECIAL_CHAR = '@' | |
export const CHATROOM_CHANNEL_MENTION_SPECIAL_CHAR = '#' | ||
|
||
// chatroom events | ||
export const CHATROOM_MESSAGE_ACTION = 'chatroom-message-action' | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We don't use this action anymore unless we use the contracts of old versions. I remember this action was removed in the PR #1521 where @corrideat implemented E2E Protocol. |
||
export const MESSAGE_RECEIVE = 'message-receive' | ||
export const MESSAGE_SEND = 'message-send' | ||
|
||
|
@@ -101,15 +101,6 @@ export const MESSAGE_VARIANTS = { | |
FAILED: 'failed' | ||
} | ||
|
||
export const PROPOSAL_VARIANTS = { | ||
CREATED: 'created', | ||
EXPIRING: 'expiring', | ||
ACCEPTED: 'accepted', | ||
REJECTED: 'rejected', | ||
CANCELLED: 'cancelled', | ||
EXPIRED: 'expired' | ||
} | ||
|
||
export const MESSAGE_NOTIFY_SETTINGS = { | ||
ALL_MESSAGES: 'all-messages', | ||
DIRECT_MESSAGES: 'direct-messages', | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -5,8 +5,12 @@ import { | |
object, string, optional, number, mapOf, literalOf | ||
} from '~/frontend/model/contracts/misc/flowTyper.js' | ||
import { | ||
CHATROOM_TYPES, CHATROOM_PRIVACY_LEVEL, | ||
MESSAGE_TYPES, MESSAGE_NOTIFICATIONS, PROPOSAL_VARIANTS, POLL_TYPES | ||
CHATROOM_TYPES, | ||
CHATROOM_PRIVACY_LEVEL, | ||
MESSAGE_TYPES, | ||
MESSAGE_NOTIFICATIONS, | ||
POLL_TYPES, | ||
STATUS_EXPIRING | ||
} from './constants.js' | ||
|
||
// group.js related | ||
|
@@ -31,14 +35,14 @@ export const chatRoomAttributesType: any = objectOf({ | |
|
||
export const messageType: any = objectMaybeOf({ | ||
type: unionOf(...Object.values(MESSAGE_TYPES).map(v => literalOf(v))), | ||
text: string, // message text | notificationType when type if NOTIFICATION | ||
text: string, | ||
proposal: objectMaybeOf({ | ||
proposalId: string, | ||
proposalType: string, | ||
expires_date_ms: number, | ||
createdDate: string, | ||
creatorID: string, | ||
variant: unionOf(...Object.values(PROPOSAL_VARIANTS).map(v => literalOf(v))) | ||
variant: unionOf([STATUS_EXPIRING].map(v => literalOf(v))) // NOTE: only expiring proposals could be notified at the moment | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @Silver-IT Please feel free to add all of the types of proposals here, as we want to be able to notify the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I was planning to add all types while I would be working on Issue #2138 after this PR is merged. Do you want me to do it in this PR? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. OK, we can save it for a separate PR 👍 |
||
}), | ||
notification: objectMaybeOf({ | ||
type: unionOf(...Object.values(MESSAGE_NOTIFICATIONS).map(v => literalOf(v))), | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -21,8 +21,8 @@ sbp('sbp/selectors/register', { | |
|
||
// Creates the notification object in a single step. | ||
const notification = { | ||
avatarUserID: template.avatarUserID || sbp('state/vuex/getters').ourIdentityContractId, | ||
...template, | ||
avatarUserID: template.avatarUserID || sbp('state/vuex/getters').ourIdentityContractId, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Updated order so that |
||
// Sets 'groupID' if this notification only pertains to a certain group. | ||
...(template.scope === 'group' ? { groupID: data.groupID } : {}), | ||
read: false, | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To make
data
meaningful, I've updated the type fromarray
toobject
.