データ型が選択リスト値の項目がありまして、リスト値の数が200くらいあるものを五十音順に変更しようと思っています。漢字・アルファベットが混在するリストなので文字コード順に表示では完全に五十音順に変更できなくて困っています。エクセルでリストをPHNETIC関数で五十音にソートできたのですが、この順番をデータローダ等で一括で順番変更できないでしょうか。
私の回答としてはメタデータをワークベンチを使って一括更新するとしましたが、さらに調べてみると他にもVisual Studio Code + Salesforce CLIも使えるかもしれません。
Tooling APIを使う方法もありそうです。
ステップ 1: 選択リストのメタデータと値を取得する
最初のステップは、DurableId選択リストフィールドのを取得することです。この使用例では、Salesforce Query アクティビティEntityDefinitionを使用して、次のサブクエリを使用してオブジェクトをクエリします
テスト用に作ったCustomObjectを指定して実行してみます。カスタムオブジェクトなので__cを忘れずに... 私は忘れてしまって表示されずにしばし悩みました。
SELECT (SELECT Id, DeveloperName, DurableId, QualifiedApiName, Label, DataType FROM Fields WHERE DataType = 'Picklist'), DeveloperName, QualifiedApiName, NewUrl FROM EntityDefinition WHERE QualifiedApiName = 'CustomObject__c'
[{"Id":"000000000000000AAA","DeveloperName":"myPicklist","DurableId":"01I5i0000034UoY.00N5i00000UNAbr","QualifiedApiName":"myPicklist__c","Label":"myPicklist","DataType":"Picklist"}]
以下のFieldDefinitionの説明みたいですが、どのように更新されるか理解できないですね。 HTTP PATCHで上書きする?
とりあえず、Postmanでtooling APIを使ってこの項目を表示させる
参考にした記事
ToolingAPIはコレクションにあるのですが、CustomFieldがないんですよね。タブを追加して作ってしまいましょう。(使い方がよく分かりませんが...)
パラメータのCustomFieldIdをこのページだけで指定したいのですが追加できませんでした。仕方ないので変数にします。
{{_endpoint}}/services/data/v{{version}}/tooling/sobjects/CustomField/{{CustomFieldId}}
実行すると、以下のようにこのカスタム項目の情報を取得できましたね。
ここまで簡単にできた...(ちょっと嘘です1時間悩みました)
変数を:CustomFieldIdのように書き換えたら、このページだけの変数になりましたね。
{{_endpoint}}/services/data/v{{version}}/tooling/sobjects/CustomField/:CustomFieldId
ちゃんと値もセットできました。
メッソドのPATCHを作って送信してみる
先程得たJsonをコピーして、ラベルをA1,B1,C1に変えてみます。
それでは送信します。
400のエラー? 不正なリクエスト?
この例を見るとBodyにJsonをセットしてPATCHとして送信しているんだけど...
To update a Tooling API object by ID, for example, MetadataContainer:
req.setEndpoint('https://MyDomainName.my.salesforce.com/services/data/v59.0/tooling/sobjects/
MetadataContainer/ + containerID + '/');
req.setBody('{"Name":"NewlyNamedContainer"}');
req.setMethod('PATCH');
Bodyのフォーマットが違うかと思って、先の記事で見たように以下の部分を削除します。
使わない感じですね。
"attributes": {
"type": "CustomField",
"url": "/services/data/v59.0/tooling/sobjects/CustomField/00N5i00000UNAbrEAH"
},
"Id": "00N5i00000UNAbrEAH",
"TableEnumOrId": "01I5i0000034UoYEAU",
"DeveloperName": "myPicklist",
"Description": null,
"Length": null,
"Precision": null,
"Scale": null,
"RelationshipLabel": null,
"SummaryOperation": null,
"InlineHelpText": null,
"MaskType": null,
"MaskChar": null,
"NamespacePrefix": null,
"ManageableState": "unmanaged",
"CreatedDate": "2023-12-04T01:49:46.000+0000",
"CreatedById": "0055i00000BXxguAAD",
"LastModifiedDate": "2023-12-04T06:37:17.000+0000",
"LastModifiedById": "0055i00000BXxguAAD",
"EntityDefinitionId": "01I5i0000034UoY",
エラーの詳細が分からない
メイン画面のいろんなところをクリックしていると... あったエラーの詳細があったよ。
[
{
"message": "Cannot deserialize instance of complexvalue from VALUE_NULL value null or request may be missing a required field",
"errorCode": "JSON_PARSER_ERROR"
}
]
意味が分からん。
complexvalue のインスタンスを VALUE_NULL 値 null から逆シリアル化できないか、リクエストに必須フィールドが欠落している可能性があります
そういえば、最初に読んだ記事では意味不明で読み飛ばしたけど"valueSettings": null'を'"valueSettings": []に置き換えていそうです。
As a transformation will not generate an empty array, a script is used to replace the null value returned in the valueSettings field with an empty array:
$io = Replace($io,'"valueSettings": null','"valueSettings": []');
この部分かな?
違うエラーになりましたね。
リターンする内容はないということかな?
サイド取得したら、見事にラベルが変更できてました。
Apexでコードを書くよりかなり楽ですね。
オブジェクトマネージャーからも確認できました。
重要なこと 今定義されている全ての選択値を再度更新しないと、非アクティブ化されてしまう。
This is the standard behavior with picklists. When you perform a deployment against the picklist, the value set is updated to match the deployed picklist values, and any values existing in the org that are not present in the deployment are deactivated.
There is no way to change this behavior or to address individual picklist value set entries.
Instead, you must retrieve the picklist value set first, modify it to add the entries you desire, and then deploy the entire value set back into the org.
これは選択リストの標準的な動作です。選択リストに対してリリースを実行すると、値セットはデプロイされた選択リストの値と一致するように更新され、組織内に存在し、リリースに存在しない値はすべて非アクティブ化されます。
この動作を変更したり、個別の選択リスト値セットのエントリに対処したりする方法はありません。
代わりに、最初に選択リスト値セットを取得し、それを変更し
値も変えてみる
既にA1(API名はA)のレコードがある状態でAPI名をA2に変更してみる。
以下のエラーになります。
[
{
"message": "Duplicate label: A1",
"errorCode": "FIELD_INTEGRITY_EXCEPTION",
"fields": []
}
]
その選択リスト値の API 参照名に変更があるグローバル選択リスト値セットを含む変更セットをリリースしようとすると、「ラベルの重複」エラーが発生する場合があります。
これは、期待される動作とみなしてください。
変更セットを介して選択リスト値の変更をリリースすると、システムは選択リスト値の API 参照名をその一意識別子として参照します。そのため、選択リスト値のラベルのみを変更しようとした場合、システムはその API 参照名に基づいて値を認識し、変更を行います。
ただし、API 参照名自体を変更しようとした場合、システムはその意図を、新しい選択リスト値を作成しようとしたものと判断し、同じラベルを持つ値がシステムにすでに存在していることを認識し、結果としてラベル重複エラーをスローします。
ということは、API名は変更できないってこと?
まだ使ているレコードが無いAPI名BをB1に変えるけど、API名は変更できませんね。ラベルだけかな?