Partager via


Geocode Dataflow Sample Code

Note

Bing Maps Spatial Data Service Geocode Dataflow API retirement

Bing Maps Spatial Data Service Geocode Dataflow API is deprecated and will be retired. Free (Basic) account customers can continue to use Bing Maps Spatial Data Service Geocode Dataflow API until June 30th, 2025. Enterprise account customers can continue to use Bing Maps Spatial Data Service Geocode Dataflow API until June 30th, 2028. To avoid service disruptions, all implementations using Bing Maps Spatial Data Service Geocode Dataflow API will need to be updated to use Azure Maps Get Geocoding Batch or Azure Maps Get Reverse Geocoding Batch API by the retirement date that applies to your Bing Maps for Enterprise account type. Azure Maps Geocoding Batch API and Azure Maps Reverse Geocode Batch API will be updated to support a larger number of locations per batch soon. For detailed migration guidance, see Migrate Bing Maps Geocode Dataflow API.

Azure Maps is Microsoft's next-generation maps and geospatial services for developers. Azure Maps has many of the same features as Bing Maps for Enterprise, and more. To get started with Azure Maps, create a free Azure subscription and an Azure Maps account. For more information about azure Maps, see Azure Maps Documentation. For migration guidance, see Bing Maps Migration Overview.

The following sample code uses the Geocode Dataflow to geocode spatial data.

Notes:

  • The code uses the functionality included in Microsoft® .NET 2.0 and is written as a console application.
  • The code accepts input data in any of the formats supported by the Geocode Dataflow. Examples of supported data formats are XML and CSV(comma-separated values). See Create Job for more information about supported data formats.
  • The code processes HTTP requests returned in XML format for version 1.0 of the Geocode Dataflow schema.
  • The results of the geocode job are output to files named succeeded.txt and failed.txt.
using System;  
using System.IO;  
using System.Net;  
using System.Text;  
using System.Threading;  
using System.Xml;  
  
//This C# code sample shows how to geocode data using the Geocode Dataflow REST API.  
//For more information on this API, see the Bing Spatial Data Services SDK on MSDN:  
// https://msdn.microsoft.com/library/ff701734.aspx  
namespace GeocodeDataFlowExample  
{  
    //A summary of status information returned in the response when you check   
    // job status.   
    class DownloadDetails  
    {  
        public string jobStatus { get; set; }  
        public string suceededlink { get; set; }  
        public string failedlink { get; set; }  
  
    }  
  
    class Program  
    {  
        //Creates a geocode dataflow job and uploads spatial data to process.  
        //Parameters:   
        //   dataFilePath: The path to the file that contains the spatial data to geocode.  
        //   dataFormat: The format of the input data. Possible values are xml, csv, tab and pipe.  
        //   key: The Bing Maps Key to use for this job. The same key is used to get job status and download results.  
        //   description: Text that is used to describe the geocode dataflow job.   
        //Return value : A URL that defines the location of the geocode dataflow job that was created.  
        static string CreateJob(string dataFilePath, string dataFormat, string key, string description)  
        {  
            //Define parameters for the HTTP request  
            //  
            // The 'Content-Type' header of the HTTP Request must be "text/plain" or "application/xml"  
            // depending on the input data format.  
            //  
            string contentType = "text/plain";  
            if (dataFormat.Equals("xml", StringComparison.OrdinalIgnoreCase))  
                contentType = "application/xml";  
  
            StringBuilder queryStringBuilder = new StringBuilder();  
  
            //  
            // The 'input'(input format) and 'key' (Bing Maps Key) parameters are required.  
            //  
            queryStringBuilder.Append("input=").Append(Uri.EscapeUriString(dataFormat));  
            queryStringBuilder.Append("&");  
            queryStringBuilder.Append("key=").Append(Uri.EscapeUriString(key));  
  
            if (!String.IsNullOrEmpty(description))  
            {  
                //  
                // The 'description' parameter is optional.  
                //  
                queryStringBuilder.Append("&");  
                queryStringBuilder.Append("description=").Append(Uri.EscapeUriString(description));  
            }  
  
            //Build the HTTP URI that will upload and create the geocode dataflow job  
            UriBuilder uriBuilder = new UriBuilder("http://spatial.virtualearth.net");  
            uriBuilder.Path = "/REST/v1/dataflows/geocode";  
            uriBuilder.Query = queryStringBuilder.ToString();  
  
            //Include the data to geocode in the HTTP request  
            using (FileStream dataStream = File.OpenRead(dataFilePath))  
            {  
                HttpWebRequest request = (HttpWebRequest)WebRequest.Create(uriBuilder.Uri);  
  
                //  
                // The HTTP method must be 'POST'.  
                //  
                request.Method = "POST";  
                request.ContentType = contentType;  
  
                using (Stream requestStream = request.GetRequestStream())  
                {  
                    byte[] buffer = new byte[16384];  
                    int bytesRead = dataStream.Read(buffer, 0, buffer.Length);  
                    while (bytesRead > 0)  
                    {  
                        requestStream.Write(buffer, 0, bytesRead);  
  
                        bytesRead = dataStream.Read(buffer, 0, buffer.Length);  
                    }  
                }  
  
                //Submit the HTTP request and check if the job was created successfully.   
                using (HttpWebResponse response = (HttpWebResponse)request.GetResponse())  
                {  
                    //  
                    // If the job was created successfully, the status code should be  
                    // 201 (Created) and the 'Location' header should contain a URL  
                    // that defines the location of the new dataflow job. You use this   
                    // URL with the Bing Maps Key to query the status of your job.  
                    //  
                    if (response.StatusCode != HttpStatusCode.Created)  
                        throw new Exception ("An HTTP error status code was encountered when creating the geocode job.");  
  
                    string dataflowJobLocation = response.GetResponseHeader("Location");  
                    if (String.IsNullOrEmpty(dataflowJobLocation))  
                        throw new Exception ("The 'Location' header is missing from the HTTP response when creating a goecode job.");  
  
                    return dataflowJobLocation;  
                }  
            }  
        }  
  
        //Checks the status of a dataflow job and defines the URLs to use to download results when the job is completed.  
        //Parameters:   
        //   dataflowJobLocation: The URL to use to check status for a job.  
        //   key: The Bing Maps Key for this job. The same key is used to create the job and download results.    
        //Return value: A DownloadDetails object that contains the status of the geocode dataflow job (Completed, Pending, Aborted).  
        //              When the status is set to Completed, DownloadDetails also contains the links to download the results  
        static DownloadDetails CheckStatus(string dataflowJobLocation, string key)  
        {  
  
            DownloadDetails statusDetails = new DownloadDetails();  
            statusDetails.jobStatus = "Pending";  
  
            //Build the HTTP Request to get job status  
            UriBuilder uriBuilder = new UriBuilder(dataflowJobLocation + @"?key=" + key + "&output=xml");  
            HttpWebRequest request = (HttpWebRequest)WebRequest.Create(uriBuilder.Uri);  
  
            request.Method = "GET";  
  
            //Submit the request and read the response to get job status and to retrieve the links for   
            //  downloading the job results  
            //Note: The following conditional statements make use of the fact that the 'Status' field will    
            //  always appear after the 'Link' fields in the HTTP response.  
            using (HttpWebResponse response = (HttpWebResponse)request.GetResponse())  
            {  
  
                if (response.StatusCode != HttpStatusCode.OK)  
                    throw new Exception ("An HTTP error status code was encountered when checking job status.");  
  
                using (Stream receiveStream = response.GetResponseStream())  
                {  
                    XmlTextReader reader = new XmlTextReader(receiveStream);  
                    while (reader.Read())  
                    {  
                        if (reader.IsStartElement())  
                        {  
                            if (reader.Name.Equals("Status"))  
                            {  
                                //return job status  
                                statusDetails.jobStatus = reader.ReadString();  
  
                                return (statusDetails);  
                            }  
                            else if (reader.Name.Equals("Link"))  
                            {  
                                //Set the URL location values for retrieving   
                                // successful and failed job results  
                                reader.MoveToFirstAttribute();  
                                if (reader.Value.Equals("output"))  
                                {  
                                    reader.MoveToNextAttribute();  
                                    if (reader.Value.Equals("succeeded"))  
                                    {  
                                        statusDetails.suceededlink = reader.ReadString();  
  
                                    }  
                                    else if (reader.Value.Equals("failed"))  
                                    {  
                                        statusDetails.failedlink = reader.ReadString();  
                                    }  
                                }  
                            }  
                        }  
                    }  
  
                }  
  
            }  
            return (statusDetails);  
        }  
  
        //Downloads job results to files names Success.txt (successfully geocoded results) and   
        //   Failed.txt (info about spatial data that was not geocoded successfully).  
         //Parameters:   
        //   statusDetails: Inclues job status and the URLs to use to download all geocoded results.  
        //   key: The Bing Maps Key for this job. The same key is used to create the job and get job status.     
  
        static void DownloadResults(DownloadDetails statusDetails, string key)  
        {  
            //Write the results for data that was geocoded successfully to a file named Success.xml  
            if (statusDetails.suceededlink != null && !statusDetails.suceededlink.Equals(String.Empty))  
            {  
                //Create a request to download successfully geocoded data. You must add the Bing Maps Key to the   
                //  download location URL provided in the response to the job status request.  
                UriBuilder successUriBuilder = new UriBuilder(statusDetails.suceededlink + @"?key=" + key);  
                HttpWebRequest request1 = (HttpWebRequest)WebRequest.Create(successUriBuilder.Uri);  
  
                request1.Method = "GET";  
  
                using (HttpWebResponse response = (HttpWebResponse)request1.GetResponse())  
                {  
                    if (response.StatusCode != HttpStatusCode.OK)  
                        throw new Exception ("An HTTP error status code was encountered when downloading results.");  
  
                    using (Stream receiveStream = response.GetResponseStream())  
                    {  
                        StreamWriter successfile = new StreamWriter("Success.txt");  
                        using (StreamReader r = new StreamReader(receiveStream))  
                        {  
                            string line;  
                            while ((line = r.ReadLine()) != null)  
                            {  
                                successfile.WriteLine(line);  
                            }  
                        }  
                        successfile.Close();  
                    }  
  
                }  
            }  
  
            //If some spatial data could not be geocoded, write the error information to a file called Failed.xml  
            if (statusDetails.failedlink != null && !statusDetails.failedlink.Equals(String.Empty))  
            {  
                //Create an HTTP request to download error information. You must add the Bing Maps Key to the   
                //  download location URL provided in the response to the job status request.  
                UriBuilder failedUriBuilder = new UriBuilder(statusDetails.failedlink + @"?key=" + key);  
                HttpWebRequest request2 = (HttpWebRequest)WebRequest.Create(failedUriBuilder.Uri);  
  
                request2.Method = "GET";  
  
                using (HttpWebResponse response = (HttpWebResponse)request2.GetResponse())  
                {  
                    if (response.StatusCode != HttpStatusCode.OK)  
                        throw new Exception ("An HTTP error status code was encountered when downloading results.");  
  
                    using (Stream receiveStream = response.GetResponseStream())  
                    {  
                        StreamWriter failedfile = new StreamWriter("Failed.txt");  
                        using (StreamReader r = new StreamReader(receiveStream))  
                        {  
                            string line;  
                            while ((line = r.ReadLine()) != null)  
                            {  
                                failedfile.WriteLine(line);  
                             }  
                        }  
                        failedfile.Close();  
                    }  
  
                }  
            }  
        }  
  
        //  
        // Sample command-line:  
        // GeocodeDataFlowExample.exe <dataFilePath> <dataFormat> <key> [<description>]  
        //  
        // Where:  
        // <dataFilePath> is a path to the data file containing entities to geocode.  
        // <dataFormat> is one of these types: xml, csv, tab, pipe.  
        // <key> is a Bing Maps Key from https://www.bingmapsportal.com.  
        // <description> is an optional description for the dataflow job.  
        //  
  
        static void Main(string[] args)  
        {  
            string dataFilePath = args[0];  
            string dataFormat = args[1];  
            string key = args[2];  
            string description = null;  
  
            try  
            {  
  
                if (args.Length > 3)  
                    description = args[3];  
  
                string dataflowJobLocation = CreateJob(dataFilePath, dataFormat, key, description);  
                Console.WriteLine("Dataflow Job Location: {0}", dataflowJobLocation);  
  
                //Continue to check the dataflow job status until the job has completed  
                DownloadDetails statusDetails = new DownloadDetails();  
                do  
                {  
                    statusDetails = CheckStatus(dataflowJobLocation, key);  
                    Console.WriteLine("Dataflow Job Status: {0}", statusDetails.jobStatus);  
                    if (statusDetails.jobStatus== "Aborted")  
                        throw new Exception("Job was aborted due to an error.");  
                    Thread.Sleep(30000); //Get status every 30 seconds  
                }  
                while (statusDetails.jobStatus.Equals("Pending"));   
  
                //When the job is completed, get the results  
                //Two files are created to record the results:  
                //  Success.xml contains the data that was successfully geocoded  
                //  Failed.mxl contains the data that could not be geocoded  
  
                DownloadResults(statusDetails, key);  
  
            }  
  
            catch (Exception  e)  
            {  
                Console.WriteLine("Exception :" + e.Message);  
            }  
         }  
    }  
}