Merge remote-tracking branch 'origin/develop'

Signed-off-by: Andrey Sobolev <haiodo@gmail.com>
This commit is contained in:
Andrey Sobolev 2024-08-28 23:45:46 +07:00
commit d1096a5b0d
No known key found for this signature in database
GPG Key ID: BD80F68D68D8F7F2
13 changed files with 158 additions and 79 deletions

View File

@ -509,9 +509,9 @@ jobs:
with:
username: hardcoreeng
password: ${{ secrets.DOCKER_ACCESS_TOKEN }}
- name: Docker push staging
if: ${{ github.ref == 'refs/heads/main' }}
run: node common/scripts/install-run-rush.js docker:staging -v
# - name: Docker push staging
# if: ${{ github.ref == 'refs/heads/main' }}
# run: node common/scripts/install-run-rush.js docker:staging -v
- name: Docker push tag
if: ${{ startsWith(github.ref, 'refs/tags/v') }}
run: |

View File

@ -11,11 +11,23 @@ then
version="${a[0]}.${a[1]}.${c}-staging"
echo "Tagging stating $1 with version ${version}"
docker tag "$1:$rev_version" "$1:$version"
docker push "$1:$version"
for n in {1..5}; do
docker push "$1:$version" && break
echo 'Docker failed to push, wait 5 seconds'
sleep 5
done
else
echo "Tagging release $1 with version ${version}"
docker tag "$1:$rev_version" "$1:$version"
docker tag "$1:$rev_version" "$1:latest"
docker push "$1:$version"
docker push "$1:latest"
for n in {1..5}; do
docker push "$1:$version" && break
echo 'Docker failed to push, wait 5 seconds'
sleep 5
done
for n in {1..5}; do
docker push "$1:latest" && break
echo 'Docker failed to push, wait 5 seconds'
sleep 5
done
fi

View File

@ -1,4 +1,8 @@
#!/usr/bin/env bash
echo "Tagging release $1 with version $2"
docker tag "$1" "$1:$2"
docker push "$1:$2"
for n in {1..5}; do
docker push "$1:$2" && break
echo 'Docker failed to push, wait 5 seconds'
sleep 5
done

View File

@ -32,12 +32,11 @@ import {
import attachment from '@hcengineering/model-attachment'
import contact from '@hcengineering/model-contact'
import core, { TAttachedDoc, TDoc } from '@hcengineering/model-core'
import notification from '@hcengineering/model-notification'
import view, { createAction } from '@hcengineering/model-view'
import setting from '@hcengineering/setting'
import love from '@hcengineering/model-love'
import gmail from './plugin'
import { defineNotifications } from './notification'
export { gmailId } from '@hcengineering/gmail'
export { gmailOperation } from './migration'
@ -215,32 +214,6 @@ export function createModel (builder: Builder): void {
gmail.action.WriteEmail
)
builder.createDoc(
notification.class.NotificationGroup,
core.space.Model,
{
label: gmail.string.Email,
icon: contact.icon.Email
},
gmail.ids.EmailNotificationGroup
)
builder.createDoc(
notification.class.NotificationType,
core.space.Model,
{
label: gmail.string.NewMessage,
generated: false,
hidden: false,
txClasses: [core.class.TxCreateDoc],
objectClass: gmail.class.Message,
group: gmail.ids.EmailNotificationGroup,
allowedForAuthor: true,
defaultEnabled: false
},
gmail.ids.EmailNotification
)
builder.mixin(gmail.class.Message, core.class.Class, core.mixin.FullTextSearchContext, {
parentPropagate: false
})
@ -259,35 +232,9 @@ export function createModel (builder: Builder): void {
]
})
builder.createDoc(
notification.class.NotificationProvider,
core.space.Model,
{
icon: contact.icon.Email,
label: gmail.string.Email,
description: gmail.string.EmailNotificationsDescription,
defaultEnabled: true,
canDisable: true,
depends: notification.providers.InboxNotificationProvider,
order: 300
},
gmail.providers.EmailNotificationProvider
)
builder.createDoc(notification.class.NotificationProviderDefaults, core.space.Model, {
provider: notification.providers.InboxNotificationProvider,
ignoredTypes: [],
enabledTypes: [gmail.ids.EmailNotification]
builder.mixin(gmail.class.Message, core.class.Class, view.mixin.ObjectTitle, {
titleProvider: gmail.function.MessageTitleProvider
})
builder.createDoc(notification.class.NotificationProviderDefaults, core.space.Model, {
provider: gmail.providers.EmailNotificationProvider,
ignoredTypes: [
gmail.ids.EmailNotification,
notification.ids.CollaboratoAddNotification,
love.ids.InviteNotification,
love.ids.KnockNotification
],
enabledTypes: []
})
defineNotifications(builder)
}

View File

@ -0,0 +1,82 @@
//
// Copyright © 2024 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
//
import notification from '@hcengineering/model-notification'
import core from '@hcengineering/model-core'
import contact from '@hcengineering/model-contact'
import love from '@hcengineering/model-love'
import { type Builder } from '@hcengineering/model'
import gmail from './plugin'
export function defineNotifications (builder: Builder): void {
builder.createDoc(
notification.class.NotificationGroup,
core.space.Model,
{
label: gmail.string.Email,
icon: contact.icon.Email
},
gmail.ids.EmailNotificationGroup
)
builder.createDoc(
notification.class.NotificationType,
core.space.Model,
{
label: gmail.string.NewMessage,
generated: false,
hidden: false,
txClasses: [core.class.TxCreateDoc],
objectClass: gmail.class.Message,
group: gmail.ids.EmailNotificationGroup,
allowedForAuthor: true,
defaultEnabled: false
},
gmail.ids.EmailNotification
)
builder.createDoc(
notification.class.NotificationProvider,
core.space.Model,
{
icon: contact.icon.Email,
label: gmail.string.Email,
description: gmail.string.EmailNotificationsDescription,
defaultEnabled: true,
canDisable: true,
depends: notification.providers.InboxNotificationProvider,
order: 300
},
gmail.providers.EmailNotificationProvider
)
builder.createDoc(notification.class.NotificationProviderDefaults, core.space.Model, {
provider: notification.providers.InboxNotificationProvider,
ignoredTypes: [],
enabledTypes: [gmail.ids.EmailNotification]
})
builder.createDoc(notification.class.NotificationProviderDefaults, core.space.Model, {
provider: gmail.providers.EmailNotificationProvider,
ignoredTypes: [
gmail.ids.EmailNotification,
notification.ids.CollaboratoAddNotification,
love.ids.InviteNotification,
love.ids.KnockNotification
],
enabledTypes: []
})
}

View File

@ -14,7 +14,7 @@
// limitations under the License.
//
import { type Doc, type Ref } from '@hcengineering/core'
import { type Client, type Doc, type Ref } from '@hcengineering/core'
import { type IntlString, mergeIds, type Resource } from '@hcengineering/platform'
import { gmailId } from '@hcengineering/gmail'
import gmail from '@hcengineering/gmail-resources/src/plugin'
@ -52,6 +52,7 @@ export default mergeIds(gmailId, gmail, {
GmailWriteMessage: '' as AnyComponent
},
function: {
HasEmail: '' as Resource<(doc?: Doc | Doc[] | undefined) => Promise<boolean>>
HasEmail: '' as Resource<(doc?: Doc | Doc[] | undefined) => Promise<boolean>>,
MessageTitleProvider: '' as Resource<(client: Client, ref: Ref<Doc>, doc?: Doc) => Promise<string>>
}
})

View File

@ -177,7 +177,7 @@
<MessageTimestamp date={message.createdOn ?? message.modifiedOn} shortTime />
</span>
{:else}
<div class="min-w-6 mt-1 relative flex-no-shrink">
<div class="avatar mt-1 relative flex-no-shrink">
{#if $$slots.icon}
<slot name="icon" />
{:else if person}
@ -344,6 +344,11 @@
}
}
.avatar {
width: 2.5rem;
height: 2.5rem;
}
.header {
display: flex;
align-items: baseline;

View File

@ -16,9 +16,15 @@
import { DisplayDocUpdateMessage, DocUpdateMessageViewlet } from '@hcengineering/activity'
import { Class, Doc, Ref } from '@hcengineering/core'
import { createQuery, getClient } from '@hcengineering/presentation'
import { Component, Icon, IconAdd, IconDelete } from '@hcengineering/ui'
import view from '@hcengineering/view'
import { buildRemovedDoc, checkIsObjectRemoved, DocNavLink, getDocLinkTitle } from '@hcengineering/view-resources'
import { AnyComponent, Component, Icon, IconAdd, IconDelete } from '@hcengineering/ui'
import view, { ObjectPanel } from '@hcengineering/view'
import {
buildRemovedDoc,
checkIsObjectRemoved,
DocNavLink,
getDocLinkTitle,
isAttachedDoc
} from '@hcengineering/view-resources'
export let objectClass: DisplayDocUpdateMessage['objectClass']
export let objectId: DisplayDocUpdateMessage['objectId']
@ -59,6 +65,18 @@
}
$: void loadObject(objectId, objectClass)
function getPanelComponent (object: Doc, objectPanel?: ObjectPanel): AnyComponent {
if (objectPanel !== undefined) {
return objectPanel.component
}
if (isAttachedDoc(object)) {
return view.component.AttachedDocPanel
}
return view.component.EditDoc
}
</script>
{#if object}
@ -84,7 +102,7 @@
{object}
colorInherit
disabled={action === 'remove'}
component={objectPanel?.component ?? view.component.EditDoc}
component={getPanelComponent(object, objectPanel)}
shrink={0}
>
<span class="overflow-label select-text">{value}</span>

View File

@ -14,7 +14,7 @@
-->
<script lang="ts">
import contact, { Channel } from '@hcengineering/contact'
import { Class, Ref } from '@hcengineering/core'
import { Class, Doc, Ref } from '@hcengineering/core'
import { createQuery, getClient } from '@hcengineering/presentation'
import { AnyComponent, Component } from '@hcengineering/ui'
import { channelProviders } from '../utils'
@ -23,6 +23,7 @@
export let _id: Ref<Channel>
export let _class: Ref<Class<Channel>>
export let embedded: boolean = false
export let selectedDoc: Ref<Doc> | undefined = undefined
export let activityMessage: DocUpdateMessage | undefined = undefined
const client = getClient()
@ -52,7 +53,7 @@
_id: channel?.attachedTo,
_class: channel?.attachedToClass,
channel,
messageId: activityMessage?.objectId
messageId: selectedDoc ?? activityMessage?.objectId
}}
on:close
/>

View File

@ -25,7 +25,7 @@ import IconGmail from './components/icons/GmailColor.svelte'
import Main from './components/Main.svelte'
import NewMessages from './components/NewMessages.svelte'
import gmail from '@hcengineering/gmail'
import { checkHasEmail } from './utils'
import { checkHasEmail, MessageTitleProvider } from './utils'
export default async (): Promise<Resources> => ({
component: {
@ -40,7 +40,8 @@ export default async (): Promise<Resources> => ({
GmailSharedMessage
},
function: {
HasEmail: checkHasEmail
HasEmail: checkHasEmail,
MessageTitleProvider
},
handler: {
DisconnectHandler: async () => {

View File

@ -5,7 +5,7 @@ import contact, {
type PersonAccount,
getName as getContactName
} from '@hcengineering/contact'
import { type Doc, type IdMap, type Ref, toIdMap } from '@hcengineering/core'
import { type Client, type Doc, type IdMap, type Ref, toIdMap } from '@hcengineering/core'
import { type Message, type SharedMessage } from '@hcengineering/gmail'
import { getClient } from '@hcengineering/presentation'
import gmail from './plugin'
@ -129,3 +129,9 @@ export function getName (
return emp != null ? `${getContactName(h, emp)} (${emailVal})` : emailVal
}
}
export async function MessageTitleProvider (client: Client, ref: Ref<Message>, doc?: Message): Promise<string> {
const object = doc ?? (await client.findOne(gmail.class.Message, { _id: ref }))
return object?.subject ?? ''
}

View File

@ -40,7 +40,7 @@
{#if doc && panelComponent}
<Component
is={panelComponent}
props={{ embedded, _id: doc.attachedTo, _class: doc.attachedToClass, ...props }}
props={{ embedded, _id: doc.attachedTo, _class: doc.attachedToClass, selectedDoc: _id, ...props }}
on:close
/>
{/if}

View File

@ -249,7 +249,7 @@ export class SpaceSecurityMiddleware extends BaseMiddleware implements Middlewar
params: null
}
ctx.derived.txes.push(tx)
ctx.derived.targets.security = (it) => {
ctx.derived.targets['security' + tx._id] = (it) => {
// TODO: I'm not sure it is called
if (it._id === tx._id) {
return targets
@ -383,7 +383,9 @@ export class SpaceSecurityMiddleware extends BaseMiddleware implements Middlewar
await this.processTx(ctx, tx)
const res = await this.provideTx(ctx, tx)
for (const txd of ctx.derived.txes) {
await this.processTx(ctx, txd)
if (txd._id !== tx._id) {
await this.processTx(ctx, txd)
}
}
return res
}