Как отправить письмо клиенту от имени сотрудника
Scope:
crmКто может выполнять метод: пользователи с правом на изменение элемента CRM
Если вы разрабатываете интеграции для Битрикс24 с помощью AI-инструментов (Codex, Claude Code, Cursor), подключите MCP-сервер, чтобы ассистент использовал официальную REST-документацию.
Письмо клиенту можно отправить автоматически через CRM. В поле «От кого» будут указаны имя и адрес электронной почты сотрудника. В карточке контакта добавится событие для исходящего письма.
Чтобы отправить письмо, последовательно выполним три метода:
-
crm.contact.get — получим данные клиента
-
user.get— получим данные сотрудника
-
crm.activity.add — создадим дело типа «Письмо»
1. Получим данные клиента
Используем метод crm.contact.get с идентификатором клиента. Значение идентификатора можно предварительно сохранить в переменной contactID. Например, получим данные контакта с идентификатором 1.
Как использовать примеры в документации
let contactID = 1;
BX24.callMethod(
'crm.contact.get',
{ 'id': contactID },
function(result) {
if (result.error()) {
reject(result.error());
} else {
resolve(result.data());
}
}
);
$contactID = 1;
$resultContact = CRest::call(
'crm.contact.get',
[
'id' => $contactID
]
);
from b24pysdk import BitrixWebhook, Client
client = Client(
BitrixWebhook(
domain="your-domain.bitrix24.com",
webhook_token="user_id/webhook_key",
)
)
contact_id = 1
result_contact = client.crm.contact.get(
bitrix_id=contact_id,
).response.result
В результате получим данные клиента, включая адрес электронной почты EMAIL и идентификатор ответственного сотрудника ASSIGNED_BY_ID.
{
"result": {
"ID": "1",
"NAME": "Алексей",
"SECOND_NAME": "Кириллович",
"LAST_NAME": "Вронский",
"ASSIGNED_BY_ID": "61",
"EMAIL": [
{
"ID": "1328",
"VALUE_TYPE": "WORK",
"VALUE": "vronsky@example.ru",
"TYPE_ID": "EMAIL"
}
]
}
}
2. Получим данные сотрудника
Чтобы получить данные ответственного сотрудника, используем метод user.get с фильтром по идентификатору сотрудника. Идентификатор должен принимать значение из поля ASSIGNED_BY_ID объекта resultContact.
BX24.callMethod(
'user.get',
{
'filter': {
'ID': resultContact.ASSIGNED_BY_ID
}
},
function(result) {
if (result.error()) {
reject(result.error());
} else {
resolve(result.data());
}
}
);
{
$resultUser = CRest::call(
'user.get',
[
'filter' => [
'ID' => $resultContact['result']['ASSIGNED_BY_ID']
]
]
);
}
result_user = client.user.get(
filter={
"ID": result_contact["ASSIGNED_BY_ID"],
}
).response.result
Получим данные сотрудника, включая адрес электронной почты EMAIL.
{
"result": [
{
"ID": "61",
"ACTIVE": true,
"NAME": "Иван",
"LAST_NAME": "Петров",
"EMAIL": "ivanpetrov@example.ru"
}
]
}
3. Создадим дело типа «Письмо»
Подготовим переменные:
-
contactEmail— первый элемент из контактаresultContact, -
staff— первый элемент из объектаresultUser.
let contactEmail = resultContact.EMAIL[0];
let staff = resultUser[0];
$contactEmail = reset($resultContact['result']['EMAIL']);
$staff = reset($resultUser['result']);
contact_email = result_contact["EMAIL"][0]
staff = result_user[0]
Чтобы добавить событие и отправить письмо, используем метод crm.activity.add. В него нужно передать данные клиента, сотрудника и параметры дела.
-
SUBJECT— тема письма. Укажемsubject email now. -
DESCRIPTION— текст письма. Например,body email now. -
DESCRIPTION_TYPE— тип текста. Возможные значения:1— обычный текст,2— HTML-разметка,3— BB-код. Зададим значение3. -
COMPLETED— флаг показывает, завершено ли событие. УкажемY. -
DIRECTION— направление активности. Передаем2— исходящее письмо. Полный список направлений активности можно получить с помощью метода crm.enum.activitydirection. -
OWNER_ID— идентификатор контакта. Передаем переменнуюcontactID. -
OWNER_TYPE_ID— идентификатор типа объекта CRM. Передаем3— контакт. Полный список типов объектов можно получить с помощью метода crm.enum.ownertype. -
TYPE_ID— тип активности. Укажем4— письмо. Список типов активности можно получить с помощью метода crm.enum.activitytype. -
COMMUNICATIONS— контактные данные клиента:-
VALUE— адрес электронной почты, берем значениеVALUEиз массиваcontactEmail, -
ENTITY_ID— идентификатор клиента, передаемcontactID, -
ENTITY_TYPE_ID— идентификатор типа объекта, передаем3— контакт.
-
-
START_TIMEиEND_TIME— дата и время начала и окончания активности. Укажем длительность 1 час. -
RESPONSIBLE_ID— идентификатор ответственного, передаемstaff.ID. -
SETTINGS— дополнительные настройки:MESSAGE_FROM— отправитель письма, передаем имяstaff.NAME, фамилиюstaff.LAST_NAMEи адрес электронной почтыstaff.EMAILсотрудника.
BX24.callMethod(
'crm.activity.add',
{
'fields': {
"SUBJECT": "subject email now",
"DESCRIPTION": "body email now",
"DESCRIPTION_TYPE": 3,
"COMPLETED": "Y",
"DIRECTION": 2,
"OWNER_ID": contactID,
"OWNER_TYPE_ID": 3,
"TYPE_ID": 4,
"COMMUNICATIONS": [
{
'VALUE': contactEmail.VALUE,
'ENTITY_ID': contactID,
'ENTITY_TYPE_ID': 3
}
],
"START_TIME": new Date().toISOString(),
"END_TIME": new Date(Date.now() + 3600 * 1000).toISOString(),
"RESPONSIBLE_ID": staff.ID,
'SETTINGS': {
'MESSAGE_FROM': `${staff.NAME} ${staff.LAST_NAME} <${staff.EMAIL}>`
}
}
}
);
$resultActivity = CRest::call(
'crm.activity.add',
[
'fields' => [
"SUBJECT" => "subject email now",
"DESCRIPTION" => "body email now",
"DESCRIPTION_TYPE" => 3,//text,html,bbCode type id in: CRest::call('crm.enum.contenttype');
"COMPLETED" => "Y",//send now
"DIRECTION" => 2,// CRest::call('crm.enum.activitydirection');
"OWNER_ID" => $contactID,
"OWNER_TYPE_ID" => 3, // CRest::call('crm.enum.ownertype');
"TYPE_ID" => 4, // CRest::call('crm.enum.activitytype');
"COMMUNICATIONS" => [
[
'VALUE' => $contactEmail['VALUE'],
'ENTITY_ID' => $contactID,
'ENTITY_TYPE_ID' => 3// CRest::call('crm.enum.ownertype');
]
],
"START_TIME" => date("Y-m-d H:i:s", time()),
"END_TIME" => date("Y-m-d H:i:s", time() + 3600),
"RESPONSIBLE_ID" => $staff['ID'],
'SETTINGS' => [
'MESSAGE_FROM' => implode(
' ',
[$staff['NAME'], $staff['LAST_NAME'], '<' . $staff['EMAIL'] . '>']
),
],
]
]
);
from datetime import datetime, timedelta
contact_email = result_contact["EMAIL"][0]
staff = result_user[0]
now = datetime.now()
result_activity = client.crm.activity.add(
fields={
"SUBJECT": "subject email now",
"DESCRIPTION": "body email now",
"DESCRIPTION_TYPE": 3,
"COMPLETED": "Y",
"DIRECTION": 2,
"OWNER_ID": contact_id,
"OWNER_TYPE_ID": 3,
"TYPE_ID": 4,
"COMMUNICATIONS": [
{
"VALUE": contact_email["VALUE"],
"ENTITY_ID": contact_id,
"ENTITY_TYPE_ID": 3,
}
],
"START_TIME": now.isoformat(timespec="seconds"),
"END_TIME": (now + timedelta(hours=1)).isoformat(timespec="seconds"),
"RESPONSIBLE_ID": staff["ID"],
"SETTINGS": {
"MESSAGE_FROM": f"{staff['NAME']} {staff['LAST_NAME']} <{staff['EMAIL']}>"
},
}
).response.result
Если событие создано успешно, метод вернет его идентификатор. Если вы получили ошибку error, изучите описание возможных ошибок в документации метода crm.activity.add.
{
"result": 3165,
}
Полный пример кода
Код в примере объединяет все шаги: получает данные клиента и сотрудника, добавляет дело «Письмо» и отправляет письмо клиенту.
document.addEventListener('DOMContentLoaded', function() {
async function createEmailActivityForContact() {
try {
let contactID = 1;
let resultContact = await new Promise((resolve, reject) => {
BX24.callMethod(
'crm.contact.get',
{ 'id': contactID },
function(result) {
if (result.error()) {
reject(result.error());
} else {
resolve(result.data());
}
}
);
});
if (resultContact && resultContact.ASSIGNED_BY_ID && resultContact.EMAIL) {
let resultUser = await new Promise((resolve, reject) => {
BX24.callMethod(
'user.get',
{
'filter': {
'ID': resultContact.ASSIGNED_BY_ID
}
},
function(result) {
if (result.error()) {
reject(result.error());
} else {
resolve(result.data());
}
}
);
});
if (resultUser.length > 0) {
let contactEmail = resultContact.EMAIL[0];
let staff = resultUser[0];
if (contactEmail.VALUE && staff.EMAIL) {
let resultActivity = await new Promise((resolve, reject) => {
BX24.callMethod(
'crm.activity.add',
{
'fields': {
"SUBJECT": "subject email now",
"DESCRIPTION": "body email now",
"DESCRIPTION_TYPE": 3,
"COMPLETED": "Y",
"DIRECTION": 2,
"OWNER_ID": contactID,
"OWNER_TYPE_ID": 3,
"TYPE_ID": 4,
"COMMUNICATIONS": [
{
'VALUE': contactEmail.VALUE,
'ENTITY_ID': contactID,
'ENTITY_TYPE_ID': 3
}
],
"START_TIME": new Date().toISOString(),
"END_TIME": new Date(Date.now() + 3600 * 1000).toISOString(),
"RESPONSIBLE_ID": staff.ID,
'SETTINGS': {
'MESSAGE_FROM': `${staff.NAME} ${staff.LAST_NAME} <${staff.EMAIL}>`
}
}
},
function(result) {
if (result.error()) {
reject(result.error());
} else {
resolve(result.data());
}
}
);
});
if (resultActivity) {
console.log(JSON.stringify({ 'message': 'Activity added' }));
} else {
console.log(JSON.stringify({ 'message': 'Activity not added' }));
}
}
}
}
} catch (error) {
console.error(error);
console.log(JSON.stringify({ 'message': 'Activity not added: ' + error.message }));
}
}
createEmailActivityForContact();
});
<?
$contactID = 1;
$resultContact = CRest::call(
'crm.contact.get',
[
'id' => $contactID
]
);
$resultActivity = [];
if (!empty($resultContact['result']['ASSIGNED_BY_ID']) && !empty($resultContact['result']['EMAIL']))
{
$resultUser = CRest::call(
'user.get',
[
'filter' => [
'ID' => $resultContact['result']['ASSIGNED_BY_ID']
]
]
);
if ($resultUser['result'])
{
$contactEmail = reset($resultContact['result']['EMAIL']);
$staff = reset($resultUser['result']);
if (!empty($contactEmail['VALUE']) && !empty($staff['EMAIL']))
{
$resultActivity = CRest::call(
'crm.activity.add',
[
'fields' => [
"SUBJECT" => "subject email now",
"DESCRIPTION" => "body email now",
"DESCRIPTION_TYPE" => 3,//text,html,bbCode type id in: CRest::call('crm.enum.contenttype');
"COMPLETED" => "Y",//send now
"DIRECTION" => 2,// CRest::call('crm.enum.activitydirection');
"OWNER_ID" => $contactID,
"OWNER_TYPE_ID" => 3, // CRest::call('crm.enum.ownertype');
"TYPE_ID" => 4, // CRest::call('crm.enum.activitytype');
"COMMUNICATIONS" => [
[
'VALUE' => $contactEmail['VALUE'],
'ENTITY_ID' => $contactID,
'ENTITY_TYPE_ID' => 3// CRest::call('crm.enum.ownertype');
]
],
"START_TIME" => date("Y-m-d H:i:s", time()),
"END_TIME" => date("Y-m-d H:i:s", time() + 3600),
"RESPONSIBLE_ID" => $staff['ID'],
'SETTINGS' => [
'MESSAGE_FROM' => implode(
' ',
[$staff['NAME'], $staff['LAST_NAME'], '<' . $staff['EMAIL'] . '>']
),
],
]
]
);
}
}
}
if (!empty($resultActivity['result']))
{
echo json_encode(['message' => 'Activity add']);
}
elseif (!empty($resultActivity['error_description']))
{
echo json_encode(['message' => 'Activity not added: ' . $resultActivity['error_description']]);
}
else
{
echo json_encode(['message' => 'Activity not added']);
}
?>
from datetime import datetime, timedelta
from b24pysdk import BitrixWebhook, Client
from b24pysdk.errors import BitrixAPIError
client = Client(
BitrixWebhook(
domain="your-domain.bitrix24.com",
webhook_token="user_id/webhook_key",
)
)
contact_id = 1
try:
contact = client.crm.contact.get(bitrix_id=contact_id).response.result
result_activity = None
if contact.get("ASSIGNED_BY_ID") and contact.get("EMAIL"):
result_user = client.user.get(
filter={"ID": contact["ASSIGNED_BY_ID"]},
).response.result
if result_user:
contact_email = contact["EMAIL"][0]
staff = result_user[0]
if contact_email.get("VALUE") and staff.get("EMAIL"):
now = datetime.now()
result_activity = client.crm.activity.add(
fields={
"SUBJECT": "subject email now",
"DESCRIPTION": "body email now",
"DESCRIPTION_TYPE": 3,
"COMPLETED": "Y",
"DIRECTION": 2,
"OWNER_ID": contact_id,
"OWNER_TYPE_ID": 3,
"TYPE_ID": 4,
"COMMUNICATIONS": [
{
"VALUE": contact_email["VALUE"],
"ENTITY_ID": contact_id,
"ENTITY_TYPE_ID": 3,
}
],
"START_TIME": now.isoformat(timespec="seconds"),
"END_TIME": (now + timedelta(hours=1)).isoformat(timespec="seconds"),
"RESPONSIBLE_ID": staff["ID"],
"SETTINGS": {
"MESSAGE_FROM": f"{staff['NAME']} {staff['LAST_NAME']} <{staff['EMAIL']}>"
},
}
).response.result
if result_activity:
print({"message": "Activity add"})
else:
print({"message": "Activity not added"})
except BitrixAPIError as error:
print({"message": f"Activity not added: {error}"})