Zoho CRM Connector
Zoho CRM Connector Help
Version 7
ZappySys Logo File Version: v7
Supported Engine: 12

Zoho CRM Connector - Source Code


<?xml version="1.0" encoding="utf-8"?>
<ApiConfig Name="Zoho CRM"
  EngineVersion="12"
  Version="7"
  Id="1C3FF995-9EE6-4C55-A654-37141622704B"
	Slug="zoho-crm-connector"
	Desc="Zoho CRM Connector can be used to integrate Zoho CRM API in your App / BI Tools. You can exchange data on Accounts, Leads, Contacts and many other modules."
	HelpLink="https://www.zoho.com/crm/developer/docs/api/"
	Logo=""
	>
  <VersionHistory>
    <Change Date="2023-04-25">Show non-approved records too when you query by id (e.g. SELECT * FROM Leads WHERE Id=1234)</Change>
    <Change Date="2023-04-25">Added Id column support for DELETE (e.g. DELETE FROM Account Where Id=123)</Change>
    <Change Date="2023-04-25">Added api_name column in the output for UPSERT so when error occurs you know which field causing it</Change>
    <Change Date="2022-04-27">Added retry logic for limit reached error (status code 429)</Change>
    <Change Date="2022-01-31">Added new endpoints for Tags read/write (get_tags, get_tag_record_count, post_tags)</Change>
    <Change Date="2022-01-31">Added support for Lookup operation in SELECT, and WHERE clause for DELETE and UPDATE</Change>
    <Change Date="2021-11-02">Fixed bulk delete operation issue</Change>
    <Change Date="2021-03-15">Renamed If-Modified-Since option to ModifiedSince</Change>
    <Change Date="2021-01-15" Type="Fix">If-Modified-Since option is not working - Read data modified after some datetime</Change>
    <Change Date="2020-12-01">Initial version</Change>
  </VersionHistory>

  <ServiceUrls>
    <ServiceUrl Name="US Domain" Url="https://www.zohoapis.com/crm/v2"/>
    <ServiceUrl Name="EU Domain" Url="https://www.zohoapis.eu/crm/v2"/>
    <ServiceUrl Name="AU Domain" Url="https://www.zohoapis.com.au/crm/v2"/>
    <ServiceUrl Name="IN Domain" Url="https://www.zohoapis.in/crm/v2"/>
    <ServiceUrl Name="CN Domain" Url="https://www.zohoapis.com.cn/crm/v2"/>
  </ServiceUrls>

  <!-- https://www.zoho.com/crm/developer/docs/api/access-refresh.html -->
  <Auths>
    <Auth Type="OAuth" Desc="OAuth Connection for Zoho CRM API" HelpLink="https://www.zoho.com/crm/developer/docs/api/register-client.html"
          ConnStr="Provider=Custom;OAuthVersion=OAuth2;ScopeSeparator={space};ReturnUrl=[$RedirectUrl$];ExtraAttributesForAuthRequest=access_type=offline&amp;prompt=consent;AuthUrl=[$AccountUrl$]/oauth/v2/auth;TokenUrl=[$AccountUrl$]/oauth/v2/token;ClientId=[$ClientId$];ClientSecret=[$ClientSecret$];Scope=[$Permissions$];ExpiresInAttribute=expires_in_sec;" TestEndPoint="get_modules">
      <Params>
        <Param Name="ClientId" />
        <Param Name="ClientSecret" Secret="True" />
        <Param Name="Permissions" Value="ZohoCRM.settings.all ZohoCRM.modules.all ZohoCRM.coql.READ"/>
        <Param Name="AccountUrl" Value="https://accounts.zoho.com"
               Options="US Domain=https://accounts.zoho.com;EU Domain=https://accounts.zoho.eu;AU Domain=https://accounts.zoho.au;IN Domain=https://accounts.zoho.in;CN Domain=https://accounts.zoho.com.cn;" />
        <Param Name="RedirectUrl" Value="https://zappysys.com/oauth" Desc="This is the redirect URL you entered when you created app in Zoho Portal. URL must match exactly including trailing slash" />

        <!--below not exposed as root level property in HTTP Connection so use as placeholder-->
        <Param Name="RetryMode" Value="RetryWhenStatusCodeMatch" Options="None;RetryAny;RetryWhenStatusCodeMatch;" Hidden="True"  HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/status-codes.html" />
        <Param Name="RetryStatusCodeList" Value="429" Hidden="True" />
        <Param Name="RetryCountMax" Value="5" Hidden="True" />
        <Param Name="RetryMultiplyWaitTime" Value="True" Hidden="True" />
	
      </Params>
      <Notes>
        <![CDATA[To register custom App, perform the following steps (Detailed steps found in the help link at the end)
<ol>
  <li>Go to <a target="_blank" href="https://api-console.zoho.com/">Zoho API Console</a>
  <li>Click <b>Add Client</b> link
  <li>Select <b>Server-based Applications</b> option
  <li>Enter desired client name (Display purpose only)</b>
  <li>Enter some URL for Company homepage</b>
  <li>For Authorized Redirect URI enter https://zappysys.com/oauth (Or enter your own but we recommend using ZappySys one if possible). This URL must match on Zoho Connector UI.
  <li>Click <b>CREATE</b>.
  <li>Copy Client ID and Secret and paste on Zoho Connector UI.
</ol>
]]>
      </Notes>
    </Auth>
  </Auths>

  <Template>
    <!-- When name not specified in EndPoints node .. its considered as base for all other template -->
    <EndPoint Name="Pagination">
      <Params>
        <Param Name="Filter" Type="Property" Desc="Filter for JSON" Value="$.data[*]" />
        <Param Name="NextUrlEndIndicator" Type="Property" Value="false" />
        <Param Name="StopIndicatorAttributeOrExpr" Type="Property" Value="$.info.more_records" />
        <Param Name="PagingMode" Type="Property" Value="ByUrlParameter" />
        <Param Name="PagingByUrlAttributeName" Type="Property" Value="page" />
        <Param Name="PagingByUrlEndStrategy" Type="Property" Value="DetectBasedOnMultipleRules" />
        <Param Name="PagingEndRules" Type="Property" Value="&lt;ArrayOfPagingEndRule&gt;&lt;PagingEndRule&gt;&lt;Mode&gt;DetectBasedOnResponseStatusCode&lt;/Mode&gt;&lt;StatusCode&gt;204&lt;/StatusCode&gt;&lt;/PagingEndRule&gt;&lt;PagingEndRule&gt;&lt;Mode&gt;DetectBasedOnResponseStatusCode&lt;/Mode&gt;&lt;StatusCode&gt;304&lt;/StatusCode&gt;&lt;/PagingEndRule&gt;&lt;PagingEndRule&gt;&lt;Mode&gt;DetectBasedOnResponseStatusCode&lt;/Mode&gt;&lt;Mode&gt;DetectBasedOnRecordCount&lt;/Mode&gt;&lt;/PagingEndRule&gt;&lt;/ArrayOfPagingEndRule&gt;" />
      </Params>
    </EndPoint>
    <EndPoint Name="ModulePagination" Template="Pagination">
      <Params>
        <Param Name="approved" Type="Query" Label="Return Approved Records" Desc="To get the list of approved records. Default value is true." Options=";true;false;both"  />
        <Param Name="converted" Type="Query" Label="Return Converted Records" Desc="To retrieve the list of converted records. Default value is false." Options=";true;false;both"  />

      </Params>

      <OutputColumns>
        <!--Column Name="id" Label="Id" DataType="DT_WSTR" Length="25" /-->
        <Column Name="-Dynamic-" Expand="True" DataEndPoint="get_module_fields" DataEndPointParameters="module=[$parent.api_name$]"
                ColumnInfoMap="Name=api_name;DataType=data_type;Length=length;Raw=json_type"
                RawInfoMap="jsonobject;jsonarray"
                DataTypeMap="DT_BOOL=boolean;DT_I8=bigint;DT_I4=integer;DT_WSTR=profileimage,ownerlookup,text,picklist,phone,website,lookup;DT_DBTIMESTAMP=datetime,date;DT_R8=currency,double;DT_NTEXT=textarea"
                />

        <Column Name="Created_By.id" Label="Created_By_Id" DataType="DT_WSTR" Length="25" />
        <Column Name="Created_By.email" Label="Created_By_Email" DataType="DT_WSTR" Length="255" />
        <Column Name="Created_By.name" Label="Created_By_Name" DataType="DT_WSTR" Length="255" />

        <Column Name="Modified_By.id" Label="Modified_By_Id" DataType="DT_WSTR" Length="25" />
        <Column Name="Modified_By.email" Label="Modified_By_Email" DataType="DT_WSTR" Length="255" />
        <Column Name="Modified_By.name" Label="Modified_By_Name" DataType="DT_WSTR" Length="255" />

        <Column Name="Owner.id" Label="Owner_Id" DataType="DT_WSTR" Length="25" />
        <Column Name="Owner.email" Label="Owner_Email" DataType="DT_WSTR" Length="255" />
        <Column Name="Owner.name" Label="Owner_Name" DataType="DT_WSTR" Length="255" />

        <Column Name="$approval_state" DataType="DT_WSTR" Length="20" />
        <Column Name="$approved" DataType="DT_BOOL" />
        <Column Name="$state" DataType="DT_WSTR" Length="50" />
      </OutputColumns>
    </EndPoint>

    <EndPoint Name="CRUD" Template=".">
      <OutputColumns>
        <Column Name="details.id" Label="id" DataType="DT_WSTR" Length="255" Order="-4" />
        <Column Name="code" DataType="DT_WSTR" Length="255" Order="-3" />
        <Column Name="message" DataType="DT_WSTR" Length="1000" Order="-2" />
        <Column Name="status" DataType="DT_WSTR" Length="255" Order="-1" />
      </OutputColumns>
    </EndPoint>

    <EndPoint Name="NoPagination" Template=".">

    </EndPoint>

  </Template>

  <EndPoints>

    <EndPoint Method="POST" Name="get_module_data_coql" Template="NoPagination"
      Label="Read Data (Query Mode - Max 200 Rows Only)"
      Url="/coql" Body="{&quot;select_query&quot;: &quot;[$sql_query$]&quot;}" Filter="$.data[*]">
      <Params>
        <Param Editor="MultiLine" Name="sql_query" Desc="Your SQL query for CRM (i.e. COQL (CRM Object Query Language)). Limtation - You can fetch maximum 200 rows using COQL Query. To read all rows use Table mode. (refer to help link to learn more about COQL)"
               HelpLink="https://zappysys.com/links?url=https://www.zoho.com/crm/developer/docs/api/COQL-Overview.html"
Value="select Last_Name, First_Name, Full_Name from Contacts where Last_Name not like 'Boyle' limit 200" Functions="JSONENCODE;TRIM"/>

      </Params>
    </EndPoint>

    <EndPoint Name="get_modules" Label="List Modules" Url="/settings/modules" Template="NoPagination" Cached="True" CachedTtl="60" HelpLink="https://www.zoho.com/crm/developer/docs/api/modules-api.html">
      <Params>
        <Param Name="Filter" Type="Property" Desc="Filter for JSON" Value="$.modules[*]"
               Options="All Modules=$.modules[*];Modules With API Support=$.modules[?(@api_supported==true)];Modules Without API Support=$.modules[?(@api_supported==false)]" />
      </Params>
    </EndPoint>

    <EndPoint Name="get_territories" Label="List Territories" Url="/settings/territories" Filter="$.territories[*]" Template="NoPagination" Cached="True" CachedTtl="60" HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/territories.html">
    </EndPoint>

    <EndPoint Name="get_module_settings" Label="List Module Settings" Url="/settings/modules/[$module$]" Cached="True" CachedTtl="20" Desc="To get the metadata for a specific module. Specify the API name of the module, such as Leads, Accounts or Deals in your API request."
              Template="NoPagination" HelpLink="https://www.zoho.com/crm/developer/docs/api/modules-api.html">
      <Params>
        <Param Name="Filter" Type="Property" Desc="Filter for JSON" Value="$.modules[*]" Options="All Modules=$.modules[*];Module Profiles=$.modules[*].profiles[*];" />
        <Param Name="module" Required="True" Type="Query" Desc="Module name for which you like to get settings" Value=""
                       OptionsEndPoint="get_modules" OptionsEndPointValueColumn="api_name" OptionsEndPointLabelColumn="module_name"
                       Options="Accounts;Activities;Calls;Campaigns;Cases;Contacts;Deals;Events;Invoices;Leads;Price_Books;Products;Purchase_Orders;Quotes;Sales_Orders;Solutions;Tasks;Vendors;Your_Custom_Module_Name;"
               />
      </Params>
    </EndPoint>

    <EndPoint Name="get_tags" Group="Tags" Label="List Tags" Url="/settings/tags?per_page=2000" Cached="True" CachedTtl="20"
              Desc="List tags for module."
              Filter="$.tags[*]"
              Template="NoPagination" HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/get-tag-list.html"
      >
      <Params>
        <Param Name="module" Required="True" Type="Query" Desc="Module name for which you like to get settings" Value=""
                       OptionsEndPoint="get_modules" OptionsEndPointValueColumn="api_name" OptionsEndPointLabelColumn="module_name"
                       Options="Accounts;Activities;Calls;Campaigns;Cases;Contacts;Deals;Events;Invoices;Leads;Price_Books;Products;Purchase_Orders;Quotes;Sales_Orders;Solutions;Tasks;Vendors;Your_Custom_Module_Name;"
               />
      </Params>
      <OutputColumns>
        <Column Name="id" Label="Id" DataType="DT_WSTR" Length="25" />
        <Column Name="name" Label="Name" DataType="DT_WSTR" Length="255" />
        <Column Name="created_by.id" Label="Created_By_Id" DataType="DT_WSTR" Length="25" />
        <Column Name="created_by.name" Label="Created_By_Name" DataType="DT_WSTR" Length="255" />


        <Column Name="created_time" Label="Created_Time" DataType="DT_DBTIMESTAMP" />
        <Column Name="modified_time" Label="Modified_Time" DataType="DT_DBTIMESTAMP" />
      </OutputColumns>
    </EndPoint>

    <EndPoint Name="get_tag_record_count" Group="Tags" Label="Gets record count for specific tag for module" 
              Url="/settings/tags/[$tag_id$]/actions/records_count"
              Desc="Add tags for specified module and specified record id(s)."
              Template="NoPagination" HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/add-tags.html"
      >
      <Params>
        <Param Name="module" Type="Query" Required="True" Desc="Module name for which you like to get settings"
                       OptionsEndPoint="get_modules" OptionsEndPointValueColumn="api_name" OptionsEndPointLabelColumn="module_name"
                       Options="Accounts;Activities;Calls;Campaigns;Cases;Contacts;Deals;Events;Invoices;Leads;Price_Books;Products;Purchase_Orders;Quotes;Sales_Orders;Solutions;Tasks;Vendors;Your_Custom_Module_Name;"
               />
        <Param Name="tag_id" Required="True" Label="Tag Id" Desc="Use List Tags to findout Id for tag you like to query"
        OptionsEndPoint="get_module_tags" OptionsEndPointValueColumn="Id" OptionsEndPointLabelColumn="Name" OptionsEndPointParameters="module=[$module$]"        
      />
      </Params>
      <OutputColumns>
        <Column Name="count" Label="Count" DataType="DT_I8" />
      </OutputColumns>
    </EndPoint>

    <EndPoint Name="post_tags" Group="Tags" Label="Add / Update Tags for records" Url="[$module$]/actions/add_tags"
              Desc="Add tags for specified module and specified record id(s)."
              Filter="$.data[*]"
              Method="POST"
              MultiSelectSeparator=","
              Template="CRUD" HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/add-tags.html"
      >
      <Params>
        <Param Name="module" Required="True" Desc="Module name for which you like to get settings" Value=""
                       OptionsEndPoint="get_modules" OptionsEndPointValueColumn="api_name" OptionsEndPointLabelColumn="module_name"
                       Options="Accounts;Activities;Calls;Campaigns;Cases;Contacts;Deals;Events;Invoices;Leads;Price_Books;Products;Purchase_Orders;Quotes;Sales_Orders;Solutions;Tasks;Vendors;Your_Custom_Module_Name;"
               />
        <Param Name="ids" Type="Query" Required="True" Label="Record Id(s) (comma seperated list)" Desc="List of Record Id(s) you like to update" />
        <Param Name="tag_names" Type="Query" Required="True" Label="Tag Id" Desc="List of Tag(s) you like to add"
          MultiSelect="True" OptionsEndPoint="get_module_tags" OptionsEndPointValueColumn="Name" OptionsEndPointLabelColumn="Name" OptionsEndPointParameters="module=[$module$]" />      
        <Param Name="over_write" Label="Overwrite" Type="Query" Desc="Set this to overwrite exising tags for the record(s)" Options=";true;false"/>
      </Params>
      <OutputColumns>
        <Column Name="details.tags" Label="tags" DataType="DT_WSTR" Length="1000" Raw="True"/>
      </OutputColumns>
    </EndPoint>

    <EndPoint Name="get_module_fields" Label="List Module Fields" Url="/settings/fields" Template="NoPagination"
          Cached="True" CachedTtl="60"
          Desc="To get the field meta data for the specified module. The fields displayed are from all layouts for the module and the response does not contain layout-specific fields like mandatory fields or picklist values.">
      <Params>
        <Param Name="module" Required="True" Type="Query" Desc="Module name for which you like to get all fields" Value=""
               OptionsEndPoint="get_modules" OptionsEndPointValueColumn="api_name" OptionsEndPointLabelColumn="module_name"
               Options="Accounts;Activities;Calls;Campaigns;Cases;Contacts;Deals;Events;Invoices;Leads;Price_Books;Products;Purchase_Orders;Quotes;Sales_Orders;Solutions;Tasks;Vendors;Your_Custom_Module_Name;"
               />
        <Param Name="Filter" Type="Property" Desc="Filter for JSON" Value="$.fields[*]"
               Options="All Fields=$.fields[*];Custom Fields=$.fields[?(@custom_field==true)];System defined fields=$.fields[?(@custom_field==false)];Fields which are not readonly=$.fields[?(@read_only==false)];Readonly fields=$.fields[?(@read_only==true)]" />
      </Params>
    </EndPoint>

    <EndPoint Name="get_module_views" Label="List Views" Url="/settings/custom_views" Template="NoPagination" Cached="True" CachedTtl="60" >
      <Params>
        <Param Name="module" Required="True" Type="Query" Desc="Module name for which you like to get all views" Value=""
               OptionsEndPoint="get_modules" OptionsEndPointValueColumn="api_name" OptionsEndPointLabelColumn="module_name"
               />
        <Param Name="Filter" Type="Property" Desc="Filter for JSON" Value="$.custom_views[*]" Options="All Views=$.custom_views[*];System defined views only=$.custom_views[?(@system_defined==true)];User defined views only=$.custom_views[?(@system_defined==false)]" />
      </Params>
    </EndPoint>

    <EndPoint Name="get_[$parent.api_name$]" Group="[$parent.api_name$]" Expand="True" Template="ModulePagination" DataEndPoint="get_modules"
              ResponseFormat="Json" Label="Read [$parent.module_name$]" Url="[$parent.api_name$]"
              HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/get-records.html">
      <Params>
        <Param Name="If-Modified-Since" Type="Header"
               Label="Last Modified Date - Read data modified after it" Desc="Supply this DateTime (ISO 8601 format) - To get the list of recently modified records"
               Value="1900-12-31T00:00:00" Functions="FUN_TO_UTC_DATE" Editor="DateTime" />

        <Param Name="cvid" Type="Query" Label="Custom View ID" Desc="To get the list of records based on custom views."
            OptionsEndPoint="get_module_views" OptionEndPointParameters="module=[$parent.api_name$];"
            OptionsEndPointValueColumn="id" OptionsEndPointLabelColumn="display_value" />

        <Param Name="fields" Type="Query" Desc="To retrieve specific field values. Kepp blank to get all fields."
            OptionsEndPoint="get_module_fields" OptionEndPointParameters="module=[$parent.api_name$];"
            OptionsEndPointValueColumn="api_name" MultiSelect="True" MultiSelectSeparator="," />
			
        <Param Name="sort_by" Type="Query" Desc="Specify the field name based on which the records must be sorted."
            OptionsEndPoint="get_module_fields" OptionEndPointParameters="module=[$parent.api_name$];"
            OptionsEndPointValueColumn="api_name" />
        <Param Name="sort_order" Type="Query" Desc="To sort the list of records in either ascending or descending order." Options=";asc;desc" />
        <Param Name="territory_id" Type="Query" Desc="To get the list of records in a territory. Use get_territories endpoint to list Id and names."  />
        <Param Name="include_child" Type="Query" Desc="To include records from the child territories. Default is false."  Options=";true;false"/>
        <Param Name="converted" Type="Query" Desc="converted"  Options=";true;false;both"/>
      </Params>
      <OutputColumns>
        <Column Name="id" DataType="DT_WSTR" Length="25" Order="-1" />
      </OutputColumns>
    </EndPoint>

    <EndPoint Name="search_[$parent.api_name$]" Group="[$parent.api_name$]" Expand="True" Template="ModulePagination" DataEndPoint="get_modules"
              ResponseFormat="Json" Label="Search [$parent.module_name$]" Url="[$parent.api_name$]/search" Desc="Search records by criteria or email or phone or word or id (specify only one parameter for search and keep other blank)"
              HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/search-records.html">
      <Params>
        <Param Name="criteria" Type="Query" Label="Search By Criteria Expression" Desc="You can supply upto 10 criterias. For more complex expression use Query API. Example: ((Last_Name:equals:Burns)and(First_Name:starts_with:Ma)). You can lookup row by ID like (id:equals:1458554000067017001). Here is Syntax for expression (({api_name}:{starts_with|equals}:{value})and/or({api_name}:{starts_with|equals}:{value})) " />
        <Param Name="email" Type="Query" Label="Search By Email"  />
        <Param Name="phone" Type="Query" Label="Search By Phone"   />
        <Param Name="word" Type="Query" Label="Search By any Word" />
        <Param Name="id" Type="Query" Label="Search By Id" Key="criteria" ValueTemplate="id:equals:{$value$}" IsKey="True"/>
        <Param Name="converted" Type="Query" Options=";true;false;both" Label="Include only converted records" />
        <Param Name="approved" Type="Query" Options=";true;false;both" Value="both" Label="Include approved records" />
      </Params>
      <OutputColumns>
        <Column Name="id" DataType="DT_WSTR" Length="25" Order="-2" />
      </OutputColumns>
    </EndPoint>

    <EndPoint Name="delete_[$parent.api_name$]" Group="[$parent.api_name$]" Expand="True" Template="CRUD" DataEndPoint="get_modules"
              RequestFormat="Csv" ResponseFormat="Json" Label="Delete [$parent.module_name$]" Url="[$parent.api_name$]?ids=[$id$]" Method="DELETE"
              HelpLink="https://www.zoho.com/crm/developer/docs/api/v2/delete-records.html" Filter="$.data[*]" BatchSize="100">
      <Params>
        <Param Name="id" Label="Record ID(s) - Use Comma for multiple (e.g. 111,222)" IsKey="True" Required="True" Functions="{$rows$}"/>
        <Param Name="wf_trigger" Type="Query" Label="Fire Wordflow Trigger" Options=";true;false"/>
      </Params>

      <InputColumns>
        <Column Name="id" MapToParam="True" DataType="DT_WSTR" Length="50" Key="True"/>
      </InputColumns>

    </EndPoint>

    <EndPoint Name="post_[$parent.api_name$]" Group="[$parent.api_name$]" Expand="True" Template="CRUD" DataEndPoint="get_modules"
              Label="Insert [$parent.module_name$]" Url="[$parent.api_name$]" Method="POST"
      Filter="$.data[*]" BatchSize="100"  ContentType="application/json">

      <Body>
        <![CDATA[{
  "data": {$rows$},
  "trigger": [[$Triggers$]]
}]]>
      </Body>
      <Params>
        <Param Name="Triggers" Value="&quot;approval&quot;,&quot;workflow&quot;,&quot;blueprint&quot;"/>
      </Params>
      <OutputColumns>
        <Column Name="details.expected_data_type" Label="expected_data_type" DataType="DT_WSTR" Length="255" />
        <Column Name="details.api_name" Label="api_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_Time" Label="created_time" DataType="DT_DBTIMESTAMP" />
        <Column Name="details.Modified_Time" Label="modified_time" DataType="DT_DBTIMESTAMP" />
        <Column Name="details.Modified_By.name" Label="modified_by_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Modified_By.id" Label="modified_by_id" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_By.name" Label="created_by_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_By.id" Label="created_by_id" DataType="DT_WSTR" Length="255" />
      </OutputColumns>

      <InputColumns>
        <Column Name="-Dynamic-" Expand="True" DataEndPoint="get_module_fields" DataEndPointParameters="module=[$parent.api_name$];Filter=$.fields[?(@read_only==false)];"
                ColumnInfoMap="Name=api_name;DataType=data_type;Length=length;Raw=json_type"
                RawInfoMap="jsonobject;jsonarray"
                DataTypeMap="DT_BOOL=boolean;DT_I8=bigint;DT_I4=integer;DT_WSTR=profileimage,ownerlookup,text,picklist,phone,website,lookup;DT_DBTIMESTAMP=datetime,date;DT_R8=currency,double;DT_NTEXT=textarea"
                />
      </InputColumns>
    </EndPoint>

    <EndPoint Name="put_[$parent.api_name$]" Group="[$parent.api_name$]" Expand="True" Template="CRUD" DataEndPoint="get_modules"
               Label="Update [$parent.module_name$]" Url="[$parent.api_name$]" Method="PUT"
       Filter="$.data[*]" BatchSize="100"  ContentType="application/json">

      <Body>
        <![CDATA[{
  "data": {$rows$},
  "trigger": [[$Triggers$]]
}]]>
      </Body>
      <Params>
        <Param Name="Triggers" Value="&quot;approval&quot;,&quot;workflow&quot;,&quot;blueprint&quot;"/>
      </Params>
      <OutputColumns>
        <Column Name="details.expected_data_type" Label="expected_data_type" DataType="DT_WSTR" Length="255" />
        <Column Name="details.api_name" Label="api_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_Time" Label="created_time" DataType="DT_DBTIMESTAMP" />
        <Column Name="details.Modified_Time" Label="modified_time" DataType="DT_DBTIMESTAMP" />
        <Column Name="details.Modified_By.name" Label="modified_by_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Modified_By.id" Label="modified_by_id" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_By.name" Label="created_by_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_By.id" Label="created_by_id" DataType="DT_WSTR" Length="255" />
      </OutputColumns>

      <InputColumns>
        <Column Name="id" DataType="DT_WSTR" Length="25" Key="True" />
        <Column Name="-Dynamic-" Expand="True" DataEndPoint="get_module_fields" DataEndPointParameters="module=[$parent.api_name$];Filter=$.fields[?(@read_only==false)];"
                ColumnInfoMap="Name=api_name;DataType=data_type;Length=length;Raw=json_type"
                RawInfoMap="jsonobject;jsonarray"
                DataTypeMap="DT_BOOL=boolean;DT_I8=bigint;DT_I4=integer;DT_WSTR=profileimage,ownerlookup,text,picklist,phone,website,lookup;DT_DBTIMESTAMP=datetime,date;DT_R8=currency,double;DT_NTEXT=textarea"
                />
      </InputColumns>
    </EndPoint>

    <EndPoint Name="upsert_[$parent.api_name$]" Group="[$parent.api_name$]" Expand="True" Template="CRUD" DataEndPoint="get_modules"
               Label="Upsert [$parent.module_name$] (Update or Insert)" Url="[$parent.api_name$]/upsert" Method="POST"
       Filter="$.data[*]" BatchSize="100" ContentType="application/json">

      <Body>
        <![CDATA[{
  "data": {$rows$},
  "trigger": [[$Triggers$]],
  "duplicate_check_fields": [[$Duplicate_Check_Fields$]]
}]]>
      </Body>
      <Params>
        <Param Name="Triggers" Value="&quot;approval&quot;,&quot;workflow&quot;,&quot;blueprint&quot;"/>
        <Param Name="Duplicate_Check_Fields" Value="Email" MultiSelect="True" MultiSelectSeparator="," MultiSelectTemplate="&quot;{@value}&quot;" Options="By Email=Email;By Phone=Phone;By Mobile=Mobile"/>

      </Params>
      <OutputColumns>
        <Column Name="duplicate_field" DataType="DT_WSTR" Length="255" />
        <Column Name="details.expected_data_type" Label="expected_data_type" DataType="DT_WSTR" Length="255" />
        <Column Name="details.api_name" Label="api_name" DataType="DT_WSTR" Length="255" />		
        <Column Name="details.Modified_Time" Label="modified_time" DataType="DT_DBTIMESTAMP" />
        <Column Name="details.Modified_By.name" Label="modified_by_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Modified_By.id" Label="modified_by_id" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_Time" Label="created_time" DataType="DT_DBTIMESTAMP" />
        <Column Name="details.Created_By.name" Label="created_by_name" DataType="DT_WSTR" Length="255" />
        <Column Name="details.Created_By.id" Label="created_by_id" DataType="DT_WSTR" Length="255" />
      </OutputColumns>

      <InputColumns>
        <Column Name="-Dynamic-" Expand="True" DataEndPoint="get_module_fields" DataEndPointParameters="module=[$parent.api_name$];Filter=$.fields[?(@read_only==false)];"
                ColumnInfoMap="Name=api_name;DataType=data_type;Length=length;Raw=json_type"
                RawInfoMap="jsonobject;jsonarray"
                DataTypeMap="DT_BOOL=boolean;DT_I8=bigint;DT_I4=integer;DT_WSTR=profileimage,ownerlookup,text,picklist,phone,website,lookup;DT_DBTIMESTAMP=datetime,date;DT_R8=currency,double;DT_NTEXT=textarea"
                />
      </InputColumns>
    </EndPoint>

  </EndPoints>

  <Tables>
    <Table Name="[$parent.api_name$]" Expand="True" DataEndPoint="get_modules"
      SelectEndPoint="get_[$parent.api_name$]"
      LookupEndPoint="search_[$parent.api_name$]"
      InsertEndPoint="post_[$parent.api_name$]"
      UpdateEndPoint="put_[$parent.api_name$]"
      UpsertEndPoint="upsert_[$parent.api_name$]"
      DeleteEndPoint="delete_[$parent.api_name$]" >
    </Table>
  </Tables>
  <Examples>
    <Example Group="ODBC" Label="Read all rows" Code="SELECT * from Accounts"></Example>

    <Example Group="ODBC" Label="Update Owner of the record(s) - Account, Contacts, Deals, Leads" Desc="This examples shows how to update a special field called Owner for any module, Owner field is a JSON fragment field so you have to update like below using email id of the User. Below examples shows how to update Owner of the Account with id 1558554000137221573">
      <Code>
        <![CDATA[UPDATE Accounts 
SET Owner='{email: "bob-the-salesman@abc.com"}'
Where Id='1558554000137221573']]>
      </Code>
    </Example>

    <Example Group="ODBC" Label="Read single row by ID" Code="SELECT * from Accounts Where Id=1558554000105110008"></Example>
    <Example Group="ODBC" Label="Read reacord(s) modified after certain date" Code="SELECT * from Accounts WITH(ModifiedSince = '2020-01-07T00:00:00')"></Example>
    <Example Group="ODBC" Label="Read using COQL (Zoho Serverside Query Language)" Desc="You can write server side query so you dont have to read full data on client side. For COQL LIMIT clause must be 200 or less. For more information check this URL https://zappysys.com/links?url=https://www.zoho.com/crm/developer/docs/api/COQL-Overview.html"
             Code="SELECT * from get_module_data_coql WITH(sql_query= 'select Last_Name, First_Name, Company from Leads where Company like ''Test'' limit 200')"></Example>
    <Example Group="ODBC" Label="Update table for specific record" Code="Update Leads SET Designation='VP Sales', Company='Test' Where id=1558554000012181009"></Example>
    <Example Group="ODBC" Label="Update table for specific record (older version)" Code="Update Leads SET id='1558554000012181009' /* id must be supplied */, Designation='VP Sales', Company='Test'"></Example>
    <Example Group="ODBC" Label="Get accounts modified after certain date" Code="SELECT * from Accounts Where Account_Name LIKE 'Test%' WITH(ModifiedSince = '2020-01-07T00:00:00')"></Example>
    <Example Group="ODBC" Label="Search accounts by specific field (server side filter)"
             Desc="You can use criteria listed here https://www.zoho.com/crm/developer/docs/api/v2/search-records.html"
             Code="SELECT * FROM search_Accounts WITH(criteria='Account_Name:starts_with:test')"></Example>

    <Example Group="ODBC" Label="Create a new record" Code="INSERT INTO Accounts(Account_Name, Phone) VALUES('Company ABC','111-567-8888')"></Example>
    <Example Group="ODBC" Label="Create a new account record (with show output on / off)" Code="INSERT INTO Accounts(Account_Name, Phone) VALUES('Company ABC','111-567-8888') WITH(Output=1) "></Example>
    <Example Group="ODBC" Label="Delete single record by Id" Desc="Delete exising record by Id (single row). You can supply upto 100 comma seperated Ids" Code="DELETE FROM Accounts WHERE id=11111111111"></Example>

    <Example Group="ODBC" Label="Delete multiple records by Ids" Desc="This example shows how to delete Account Records by multiple Ids. You can supply upto 100 comma seperated Ids" Code="DELETE FROM Accounts WITH(Id='11111,22222,33333')"></Example>

	
    <Example Group="ODBC" Label="Upsert account record (Update or Insert - based on unique field(s) for module)" Code="UPSERT INTO Accounts(Account_Name, Phone) VALUES('Company ABC','111-567-8888') WITH(Output=1)"></Example>
	
    <Example Group="ODBC" Label="Upsert lead record (Update or Insert - based on unique field(s) for module)" Code="UPSERT INTO Leads(Last_Name, Email) VALUES('Patel','zpatel@abc.com') WITH(Output=1)"></Example>	

    <Example Group="ODBC" Label="Create new account(s) in BULK (read / write from external source)" Desc="This examples shows how to use SOURCE clause to read data from MS SQL Server (or other external system) and send data to Zoho using Bulk API">
      <Code>
        <![CDATA[INSERT INTO Accounts
SOURCE(
  'MSSQL' --ODBC or OLEDB
  ,'Data Source=localhost;Initial Catalog=Test;Integrated Security=true'
  ,'select ''Test Account-A'' as Account_Name,''111-111-1111'' as Phone
    UNION
    select ''Test Account-B'' as Account_Name,''222-222-2222'' as Phone
   '
)]]>
      </Code>
    </Example>

    <Example Group="ODBC" Label="UPSERT (Update or insert) account(s) in BULK (read / write from external source)" Desc="This examples shows how to use SOURCE clause to read data from MS SQL Server (or other external system) and send data to Zoho using Bulk API. Record uniqueness is checked based on Unique field setup for module (i.e. email, account name, phone etc)">
      <Code>
        <![CDATA[UPSERT INTO Accounts
SOURCE(
  'MSSQL' --ODBC or OLEDB
  ,'Data Source=localhost;Initial Catalog=Test;Integrated Security=true'
  ,'select ''Test Account-A'' as Account_Name,''111-111-1111'' as Phone
    UNION
    select ''Test Account-B'' as Account_Name,''222-222-2222'' as Phone
   '
)]]>
      </Code>
    </Example>

    <Example Group="ODBC" Label="Update Tag(s) for exising records (add or overwrite)" >
      <Code>
        <![CDATA[SELECT * FROM post_tags
WITH(
	  module='Accounts'
	, ids='1558554000105151002'
	, tag_names='mytag1,mytag2'
  , over_write='true'
)]]>
      </Code>
    </Example>
  </Examples>
</ApiConfig>