Skip to content
Snippets Groups Projects
Commit 831e1897b5e1 authored by Vincent Hatakeyama's avatar Vincent Hatakeyama
Browse files

:sparkles: block more discuss features

parent 12af9d60c479
No related branches found
No related tags found
1 merge request!2✨ block more discuss features
Pipeline #123548 passed
<?xml version="1.0" encoding="UTF-8" ?>
<svg xmlns="http://www.w3.org/2000/svg" width="114" height="20">
<linearGradient id="b" x2="0" y2="100%">
<stop offset="0" stop-color="#bbb" stop-opacity=".1" />
<stop offset="1" stop-opacity=".1" />
</linearGradient>
<mask id="anybadge_1">
<rect width="114" height="20" rx="3" fill="#fff" />
</mask>
<g mask="url(#anybadge_1)">
<path fill="#555" d="M0 0h72v20H0z" />
<path fill="#000000" d="M72 0h42v20H72z" />
<path fill="url(#b)" d="M0 0h114v20H0z" />
</g>
<g
fill="#fff"
text-anchor="middle"
font-family="DejaVu Sans,Verdana,Geneva,sans-serif"
font-size="11"
>
<text x="37.0" y="15" fill="#010101" fill-opacity=".3">code style</text>
<text x="36.0" y="14">code style</text>
</g>
<g
fill="#fff"
text-anchor="middle"
font-family="DejaVu Sans,Verdana,Geneva,sans-serif"
font-size="11"
>
<text x="94.0" y="15" fill="#010101" fill-opacity=".3">black</text>
<text x="93.0" y="14">black</text>
</g>
</svg>
<svg
xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink"
width="53"
height="20"
role="img"
aria-label="Ruff"
>
<title>Ruff</title>
<linearGradient id="s" x2="0" y2="100%">
<stop offset="0" stop-color="#bbb" stop-opacity=".1" />
<stop offset="1" stop-opacity=".1" />
</linearGradient>
<clipPath id="r">
<rect width="53" height="20" rx="3" fill="#fff" />
</clipPath>
<g clip-path="url(#r)">
<rect width="20" height="20" fill="#555" />
<rect x="20" width="33" height="20" fill="#261230" />
<rect width="53" height="20" fill="url(#s)" />
</g>
<g
fill="#fff"
text-anchor="middle"
font-family="Verdana,Geneva,DejaVu Sans,sans-serif"
text-rendering="geometricPrecision"
font-size="110"
>
<image
x="5"
y="3"
width="10"
height="14"
xlink:href=""
/>
<text
aria-hidden="true"
x="355"
y="150"
fill="#010101"
fill-opacity=".3"
transform="scale(.1)"
textLength="230"
>
Ruff
</text>
<text x="355" y="140" transform="scale(.1)" fill="#fff" textLength="230">Ruff</text>
</g>
</svg>
......@@ -2,6 +2,18 @@
Changelog
=========
17.0.1.1.0
----------
User not in the message group, can not:
- access messages related to discuss.channel
- access discuss.channel.
- see the button to display channels/messages.
If a user with discuss send a message to a user without it, the message is blocked.
The UI for that is not very nice, the message appears to be sent but is not.
17.0.1.0.0
----------
......
......@@ -17,5 +17,5 @@
|maturity| |license| |ruff| |prettier|
Introduce a new group, messaging, and only member of this group can see the messaging menu.
Introduce a new group, messaging, and only member of this group can see the discuss menu.
......@@ -21,1 +21,3 @@
Discussion of internal users is also blocked if they do not belong to this group.
from ._hook import _uninstall_hook, _post_init_hook
from . import models
###############################################################################
#
# Messaging Group a module for Odoo
# Copyright © 2014, 2018, 2022 XCG Consulting (https://xcg-consulting.fr/)
# Copyright © 2014, 2018, 2022, 2025 XCG Consulting (https://xcg-consulting.fr/)
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
......@@ -21,8 +21,8 @@
"name": "Messaging Group",
"license": "AGPL-3",
"summary": "New group to access discussions",
"version": "17.0.1.0.0",
"version": "17.0.1.1.0",
"category": "Extra Rights",
"author": "XCG Consulting",
"website": "https://orbeet.io/",
"depends": ["mail"],
......@@ -25,6 +25,11 @@
"category": "Extra Rights",
"author": "XCG Consulting",
"website": "https://orbeet.io/",
"depends": ["mail"],
"data": ["security/security.xml"],
"data": ["security/security.xml", "security/ir.model.access.csv"],
"assets": {
"web.assets_backend": [
"mail_messaging_group/static/src/discuss/**",
],
},
"auto_install": True,
......@@ -30,2 +35,4 @@
"auto_install": True,
"uninstall_hook": "_uninstall_hook",
"post_init_hook": "_post_init_hook",
}
_hook.py 0 → 100644
###############################################################################
#
# Messaging Group a module for Odoo
# Copyright © 2025 XCG Consulting (https://xcg-consulting.fr/)
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
###############################################################################
from odoo import Command # type: ignore[import-untyped]
from odoo.api import Environment # type: ignore[import-untyped]
def _uninstall_hook(env: Environment):
"""Revert changes made on other module data"""
env.ref("mail.menu_root_discuss").groups_id = [Command.clear()]
env.ref("mail.access_discuss_channel_user").active = True
env.ref("mail.channel_all_employees").group_ids = [
Command.set([env.ref("base.group_user").id])
]
def _post_init_hook(env: Environment):
"""Change some noupdate data"""
env.ref("mail.channel_all_employees").group_ids = [
Command.set([env.ref("mail_messaging_group.group_messaging").id])
]
from . import mail_message
###############################################################################
#
# Messaging Group a module for Odoo
# Copyright © 2025 XCG Consulting (https://xcg-consulting.fr/)
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
###############################################################################
from odoo import _, models # type: ignore[import-untyped]
from odoo.exceptions import AccessError # type: ignore[import-untyped]
class Message(models.Model):
_inherit = "mail.message"
def create(self, vals_list):
created = super().create(vals_list)
if not self.env.su:
for record in created:
if record.model == "discuss.channel":
channel = self.env["discuss.channel"].browse(record.res_id)
if channel.channel_type == "chat":
# check that all members have messaging group
users = (
self.sudo()
.env["res.users"]
.search(
[
(
"partner_id",
"in",
channel.mapped(
"channel_member_ids.partner_id.id"
),
)
]
)
)
for user in users:
if (
not user.has_group(
"mail_messaging_group.group_messaging"
)
and not user.has_group("base.group_public")
and not user.has_group("base.group_portal")
):
raise AccessError(
_(
"Creating messages to user that can not read "
"them is not allowed."
)
)
return created
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
access_discuss_channel_user,discuss.channel ,mail.model_discuss_channel,base.group_user,1,0,0,0
access_discuss_channel_messaging,discuss.channel messaging,mail.model_discuss_channel,group_messaging,1,1,1,0
......@@ -14,4 +14,21 @@
<record id="mail.menu_root_discuss" model="ir.ui.menu">
<field name="groups_id" eval="[(6, 0, [ref('group_messaging')])]" />
</record>
<record id="mail.access_discuss_channel_user" model="ir.model.access">
<field name="active" eval="False" />
</record>
<record id="message_no_discuss_channel_access" model="ir.rule">
<field name="name">No access to discuss messages to internal users</field>
<field name="model_id" ref="mail.model_mail_message" />
<field name="domain_force">[('model', '!=', 'discuss.channel')]</field>
<field name="groups" eval="[(4, ref('base.group_user'))]" />
</record>
<record id="message" model="ir.rule">
<field name="name">Only messaging users have access</field>
<field name="model_id" ref="mail.model_mail_message" />
<field name="domain_force">[(1, '=', 1)]</field>
<field name="groups" eval="[(4, ref('group_messaging'))]" />
</record>
</odoo>
/** @odoo-module **/
// Adapted from https://www.odoo.com/fr_FR/forum/aide-1/groups-option-on-owl-template-263872
import {patch} from "@web/core/utils/patch";
import {useService} from "@web/core/utils/hooks";
import {useState, onWillStart} from "@odoo/owl";
import {MessagingMenu} from "@mail/core/web/messaging_menu";
patch(MessagingMenu.prototype, {
setup() {
super.setup();
this.user = useService("user");
this.state = useState({
canAccessDiscuss: false, // Initial visibility state
});
// Check if the user belongs to the "mail_messaging_group.group_messaging" group
onWillStart(async () => {
this.state.canAccessDiscuss = await this.user.hasGroup(
"mail_messaging_group.group_messaging"
);
});
},
});
<?xml version="1.0" encoding="UTF-8" ?>
<templates xml:space="preserve">
<t t-inherit="mail.MessagingMenu" t-inherit-mode="extension">
<xpath expr="//Dropdown" position="attributes">
<attribute
name="t-if"
>!env.inDiscussApp &amp; state.canAccessDiscuss</attribute>
</xpath>
<xpath expr="//t[@t-else='']" position="attributes">
<attribute name="t-if">state.canAccessDiscuss</attribute>
</xpath>
</t>
</templates>
from odoo import SUPERUSER_ID, api # type: ignore[import-untyped]
def migrate(cr, _installed_version):
"""Remove user without access to messages from general channel"""
env = api.Environment(cr, SUPERUSER_ID, {})
general_channel = env.ref("mail.channel_all_employees")
if general_channel:
members_to_remove = env["discuss.channel.member"]
for member in env.ref("mail.channel_all_employees").channel_member_ids:
users = env["res.users"].search([("partner_id", "=", member.partner_id.id)])
for user in users:
if not user.has_group("mail_messaging_group.group_messaging"):
members_to_remove |= member
if members_to_remove:
members_to_remove.unlink()
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment