#salesforce #api-client #api #rust

rust_sync_force

Salesforce Sync REST API 客户端

8 个版本

0.3.2 2023 年 8 月 8 日
0.3.1 2023 年 5 月 18 日
0.2.3 2023 年 4 月 24 日
0.2.2 2023 年 3 月 31 日
0.1.0 2023 年 2 月 2 日

#1 in #salesforce

Download history 16/week @ 2024-03-12 4/week @ 2024-04-02

52 每月下载量

MIT 许可证

76KB
1.5K SLoC

crate-name at crates.io crate-name at docs.rs Rust

Rust Sync Force

Salesforce Sync 的 Rust 客户端。支持大多数 Salesforce REST 方法。也支持更改数据捕获。

Rust 异步版本在此 https://github.com/tzmfreedom/rustforce

用法

use rust_sync_force::{Client, Error};
use rust_sync_force::response::{QueryResponse, ErrorResponse};
use serde::Deserialize;
use std::env;

#[derive(Deserialize, Debug)]
#[serde(rename_all = "PascalCase")]
struct Account {
    #[serde(rename = "attributes")]
    attributes: Attribute,
    id: String,
    name: String,
}

#[derive(Deserialize, Debug)]
struct Attribute {
    url: String,
    #[serde(rename = "type")]
    sobject_type: String,
}

fn main() -> Result<(), Error> {
    
    let client_id = env::var("SFDC_CLIENT_ID").unwrap();
    let client_secret = env::var("SFDC_CLIENT_SECRET").unwrap();
    let username = env::var("SFDC_USERNAME").unwrap();
    let password = env::var("SFDC_PASSWORD").unwrap();

    let mut client = Client::new(client_id, client_secret);
    client.login_with_credential(username, password)?;

    let res: QueryResponse<Account> = client.query("SELECT Id, Name FROM Account WHERE id = '0012K00001drfGYQAY'".to_string())?;
    println!("{:?}", res);

    Ok(())
}

身份验证

用户名密码流程

let mut client = Client::new(client_id, client_secret);
client.login_with_credential(username, password)?;

[WIP]授权代码授予

刷新令牌

let r = client.refresh("xxxx")?;

查询记录

let r: Result<QueryResponse<Account>, Error> = client.query("SELECT Id, Name FROM Account")?;

查询所有记录

let r: Result<QueryResponse<Account>, Error> = client.query_all("SELECT Id, Name FROM Account")?;

按 ID 查找

let r: Result<Account, Error> = client.find_by_id("Account", "{sf_id}")?;

插入记录

let mut params = HashMap::new();
params.insert("Name", "hello rust");
let r = client.insert("Account", params)?;
println!("{:?}", r);

插入多条记录

let account1 = Account {
    name: "account name1".into(),
    attributes: Attribute { sobject_type: "Account".into() },
};

let account2 = Account {
    name: "account name2".into(),
    attributes: Attribute { sobject_type: "Account".into() },
};

let r = client.inserts(true, vec![account1, account2])?;

更新记录

let r = client.update("Account", "{sobject_id}", params)?;

更新多条记录

let account = Account {
    id: "account_id".into(),
    name: "new_name".into(),
    attributes: Attribute { sobject_type: "Account".into() },
};

let r = client.updates(true, vec![account])?;

插入或更新记录

let r = client.upsert("Account", "external_key_name", "external_key_value", params)?;

插入或更新多条记录

let account = Account {
    exkey: "external_key_id".into(),
    name: "new_name".into(),
    attributes: Attribute { sobject_type: "Account".into() },
};

let r = client.upserts(true, "Account", "ExKey__c", vec![account])?;

删除记录

let r = client.delete("Account", "{sobject_id}")?;

删除多条记录

let r = client.deletes(true, vec!["account_id".into()])?;

全局描述

let r = client.describe_global()?;

描述 SObject

let r = client.describe("Account")?;

版本

let versions = client.versions()?;

搜索(SOSL)

let r = client.search("FIND {Rust}")?;

更改数据捕获 - 流式传输

let mut client = Client::new(client_id, client_secret);
client.login_with_credential(username, password)?;

let mut stream_client = rust_sync_force::stream::CometdClient::new(
    client,

    // listen to Account Change Event
    vec!["/data/AccountChangeEvent".to_string()],
);

stream_client.init().expect("Could not init cometd client");

println!("Cometd client successfully initialized");

for response in stream_client.connect()? {
    if let StreamResponse::Delivery(delivery) = response {
        match serde_json::from_value::<SFMetadata>(delivery.data.clone()) {
            Ok(data) => {
                println!("Data: {:#?}", data);
                // Here you should have your patterns matching your own objects
            }
            Err(err) => {
                println!(
                    "SF delivery data could not be parsed: {:?}\nData:{:?}",
                    err, delivery
                )
            }
        }
    }
}

#[derive(Debug, Deserialize)]
pub struct SFChangeEventHeader {
    pub commitNumber: usize,
    pub commitUser: String,
    pub sequenceNumber: usize,
    pub entityName: String,
    pub changeType: String,
    pub commitTimestamp: usize,
    pub recordIds: Vec<String>,
}

#[derive(Debug, Deserialize)]
pub struct SFPayload {
    pub LastModifiedDate: String,
    pub ChangeEventHeader: SFChangeEventHeader,
}

#[derive(Debug, Deserialize)]
pub struct SFMetadata {
    pub schema: String,
    pub payload: SFPayload,
}

依赖关系

~6.5–9.5MB
~192K SLoC