Nota
O acesso a esta página requer autorização. Pode tentar iniciar sessão ou alterar os diretórios.
O acesso a esta página requer autorização. Pode tentar alterar os diretórios.
Este artigo demonstra o código de exemplo que usa o SDK para trabalhar com dados e metadados do Dataverse. Antes de continuar, leia Introdução.
Operações básicas
Aqui está um código de exemplo que opera na tabela da conta.
from azure.identity import InteractiveBrowserCredential
from PowerPlatform.Dataverse.client import DataverseClient
# Replace <myorg> with the name of a valid environment.
base_url = "https://<myorg>.crm.dynamics.com"
client = DataverseClient(base_url=base_url, credential=InteractiveBrowserCredential())
# Create a record
account_id = client.records.create("account", {"name": "Contoso Ltd"})
# Read a record
account = client.records.retrieve("account", account_id)
print(account["name"])
# Read with expand fetches a related record in the same HTTP request
account = client.records.retrieve(
"account", account_id,
select=["name"],
expand=["primarycontactid"],
)
contact = (account.get("primarycontactid") or {})
print(contact.get("fullname"))
# Update a record
client.records.update("account", account_id, {"telephone1": "555-0199"})
# Delete a record
client.records.delete("account", account_id)
Gerenciador de contexto
O Gerenciador de Contexto manipula a limpeza automática e o pool de conexões HTTP. Use a sintaxe a seguir para aproveitar o Gerenciador de Contexto.
with DataverseClient("https://<myorg>.crm.dynamics.com", credential) as client:
O código de trabalho a seguir demonstra como usar o Gerenciador de Contexto.
from azure.identity import InteractiveBrowserCredential
from PowerPlatform.Dataverse.client import DataverseClient
# Connect to Dataverse
credential = InteractiveBrowserCredential()
with DataverseClient("https://<myorg>.crm.dynamics.com", credential) as client:
# Create a contact
contact_id = client.records.create("contact", {"firstname": "John", "lastname": "Doe"})
# Read the contact back
contact = client.records.retrieve("contact", contact_id, select=["firstname", "lastname"])
print(f"Created: {contact['firstname']} {contact['lastname']}")
# Clean up
client.records.delete("contact", contact_id)
# Session closed, caches cleared automatically
Operações em massa
Aqui estão alguns exemplos que executam operações em massa.
# Bulk create
payloads = [
{"name": "Company A"},
{"name": "Company B"},
{"name": "Company C"}
]
ids = client.records.create("account", payloads)
# Bulk update (broadcast same change to all)
client.records.update("account", ids, {"industry": "Technology"})
# Bulk delete
client.records.delete("account", ids, use_bulk_delete=True)
O exemplo a seguir cria várias contas. Passe uma lista de cargas úteis para create(logical_name, payloads) a fim de invocar a ação associada à coleção Microsoft.Dynamics.CRM.CreateMultiple. O método retorna list[str] dos IDs de registros criados.
# Bulk create accounts (returns list of GUIDs)
payloads = [
{"name": "Contoso"},
{"name": "Fabrikam"},
{"name": "Northwind"},
]
ids = client.records.create("account", payloads)
assert isinstance(ids, list) and all(isinstance(x, str) for x in ids)
print({"created_ids": ids})
Para obter mais informações sobre operações em massa:
- Retorna
None(o mesmo que uma única atualização) para manter a semântica consistente. - A diferença entre transmissão em massa e por registro depende de o parâmetro
changesser um dicionário ou uma lista. - O atributo de chave primária é injetado automaticamente ao construir os destinos da ação UpdateMultiple.
- Se algum conteúdo omitir @odata.type, o SDK o carimba automaticamente (pesquisa de nome lógico em cache).
- A resposta inclui apenas IDs – o SDK retorna essas cadeias de caracteres GUID.
- A criação de registro único retorna uma lista de elemento único de GUIDs.
- A pesquisa de metadados para
@odata.typeé realizada uma vez por conjunto de entidades (em cache na memória).
Upsert (criar e atualizar)
Uma sequência de acesso a dados comum é primeiro verificar se existe uma linha de tabela. Se a linha existir, atualize-a. Caso contrário, crie a linha. Você pode tornar essa sequência mais eficiente usando uma única chamada de API da operação Upsert.
Para obter mais informações, consulte Usar Upsert para criar ou atualizar um registro.
Importante
A tabela deve ter uma chave alternativa configurada no Dataverse para as colunas usadas em alternate_key. Defina chaves alternativas nos metadados da tabela por meio do portal do criador de Power Apps ou de uma chamada à API do Dataverse. Sem uma chave alternativa configurada, o Dataverse rejeita solicitações upsert com um erro 400.
Use client.records.upsert() para criar ou atualizar registros identificados por chaves alternativas. Quando a chave corresponde a um registro existente, o método atualiza o registro. Caso contrário, ele criará o registro. Um item usa uma requisição PATCH, enquanto vários itens usam a ação em massa UpsertMultiple.
from PowerPlatform.Dataverse.models.upsert import UpsertItem
# Upsert a single record
client.records.upsert("account", [
UpsertItem(
alternate_key={"accountnumber": "ACC-001"},
record={"name": "Contoso Ltd", "telephone1": "555-0100"},
)
])
# Upsert multiple records (uses UpsertMultiple bulk action)
client.records.upsert("account", [
UpsertItem(
alternate_key={"accountnumber": "ACC-001"},
record={"name": "Contoso Ltd"},
),
UpsertItem(
alternate_key={"accountnumber": "ACC-002"},
record={"name": "Fabrikam Inc"},
),
])
# Composite alternate key (multiple columns identify the record)
client.records.upsert("account", [
UpsertItem(
alternate_key={"accountnumber": "ACC-001", "address1_postalcode": "98052"},
record={"name": "Contoso Ltd"},
)
])
# Plain dict syntax (no import needed)
client.records.upsert("account", [
{
"alternate_key": {"accountnumber": "ACC-001"},
"record": {"name": "Contoso Ltd"},
}
])
Estruturas de Dados (DataFrames)
O SDK fornece wrappers pandas para todas as operações CRUD por meio do client.dataframe namespace. Esses wrappers usam as APIs de DataFrame e Series do pandas para entrada e saída.
Note
client.dataframe.get() está obsoleto. Use os padrões de GA mostrados nas seções a seguir.
import pandas as pd
from PowerPlatform.Dataverse.models.filters import col
# Query records as a single DataFrame (GA builder pattern)
df = (client.query.builder("account")
.select("name", "telephone1")
.where(col("statecode") == 0)
.execute()
.to_dataframe())
print(f"Found {len(df)} accounts")
# Limit results with top for large tables
df = client.query.builder("account").select("name").top(100).execute().to_dataframe()
# Create records from a DataFrame (returns a Series of GUIDs)
new_accounts = pd.DataFrame([
{"name": "Contoso", "telephone1": "555-0100"},
{"name": "Fabrikam", "telephone1": "555-0200"},
])
new_accounts["accountid"] = client.dataframe.create("account", new_accounts)
# Update records from a DataFrame (id_column identifies the GUID column)
new_accounts["telephone1"] = ["555-0199", "555-0299"]
client.dataframe.update("account", new_accounts, id_column="accountid")
# Clear a field by setting clear_nulls=True (by default, NaN/None fields are skipped)
df = pd.DataFrame([{"accountid": new_accounts["accountid"].iloc[0], "websiteurl": None}])
client.dataframe.update("account", df, id_column="accountid", clear_nulls=True)
# Delete records by passing a Series of GUIDs
client.dataframe.delete("account", new_accounts["accountid"])
# SQL query directly to DataFrame (supports JOINs, aggregates, GROUP BY)
df = client.dataframe.sql(
"SELECT a.name, COUNT(c.contactid) as contacts "
"FROM account a "
"JOIN contact c ON a.accountid = c.parentcustomerid "
"GROUP BY a.name"
)
Carregar arquivos no Dataverse
O exemplo a seguir mostra como carregar um arquivo nomeado document.pdf para a coluna Arquivo nomeada new_Document de um registro de conta. O SDK para Python manipula automaticamente o agrupamento de arquivos para arquivos com mais de 128 MB.
# Upload a file to a record
client.files.upload(
"account",
account_id,
"new_Document",
"/path/to/document.pdf",
)
Dica
Se a coluna de arquivo não existir, o SDK a criará automaticamente.
Operações em lote
Use client.batch para enviar várias operações em uma solicitação HTTP. O namespace de lote espelha client.records, client.tables e client.query.
# Build a batch request and add operations
batch = client.batch.new()
batch.records.create("account", {"name": "Contoso"})
batch.records.create("account", [{"name": "Fabrikam"}, {"name": "Woodgrove"}])
batch.records.update("account", account_id, {"telephone1": "555-0100"})
batch.records.delete("account", old_id)
batch.records.retrieve("account", account_id, select=["name"], expand=["primarycontactid"]) # single record with expand
batch.records.list( # multi-record, single page
"account",
filter="statecode eq 0",
select=["name"],
orderby=["name asc"],
top=50,
)
result = batch.execute()
for item in result.responses:
if item.is_success:
print(f"[OK] {item.status_code} entity_id={item.entity_id}")
else:
print(f"[ERR] {item.status_code}: {item.error_message}")
Conjunto de alterações transacionais
Todas as operações em um conjunto de alterações são concluídas com êxito ou revertidas em conjunto.
batch = client.batch.new()
with batch.changeset() as cs:
lead_ref = cs.records.create("lead", {"firstname": "Ada"})
contact_ref = cs.records.create("contact", {"firstname": "Ada"})
cs.records.create("account", {
"name": "Babbage & Co.",
"originatingleadid@odata.bind": lead_ref,
"primarycontactid@odata.bind": contact_ref,
})
result = batch.execute()
print(f"Created {len(result.entity_ids)} records atomically")
Metadados de tabela e consultas SQL em um lote
batch = client.batch.new()
batch.tables.create("new_Product", {"new_Price": "decimal", "new_InStock": "bool"})
batch.tables.add_columns("new_Product", {"new_Rating": "int"})
batch.tables.get("new_Product")
batch.query.sql("SELECT TOP 5 name FROM account")
result = batch.execute()
Continuar em caso de erro
Tente todas as operações mesmo se uma falhar.
result = batch.execute(continue_on_error=True)
print(f"Succeeded: {len(result.succeeded)}, Failed: {len(result.failed)}")
for item in result.failed:
print(f"[ERR] {item.status_code}: {item.error_message}")
Integração do DataFrame
Feed pandas DataFrames diretamente em um lote.
import pandas as pd
batch = client.batch.new()
# Create records from a DataFrame
df = pd.DataFrame([{"name": "Contoso"}, {"name": "Fabrikam"}])
batch.dataframe.create("account", df)
# Update records from a DataFrame
updates = pd.DataFrame([
{"accountid": id1, "telephone1": "555-0100"},
{"accountid": id2, "telephone1": "555-0200"},
])
batch.dataframe.update("account", updates, id_column="accountid")
# Delete records from a Series
batch.dataframe.delete("account", pd.Series([id1, id2]))
result = batch.execute()
Para obter um exemplo completo em lote, consulte examples/advanced/batch.py.
Informações relacionadas
- Personalizar tabelas e colunas
- Consultar dados
- Exemplos de código do SDK para Python
- SDK para Python README