ODATA Web服务,获取$元数据C#

时间:2012-07-03 13:56:19

标签: c# metadata odata

我有一个公开此$metadata

的WebService
<?xml version="1.0" encoding="UTF-8" standalone="true"?>
<edmx:Edmx xmlns:edmx="http://schemas.microsoft.com/ado/2007/06/edmx" Version="1.0">     > <edmx:DataServices m:DataServiceVersion="1.0"
xmlns:m="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata">

<Schema xmlns:m="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata"
xmlns="http://schemas.microsoft.com/ado/2007/05/edm"
xmlns:d="http://schemas.microsoft.com/ado/2007/08/dataservices"
Namespace="NAV"> 
<EntityType Name="PAsset"> 
<Key> <PropertyRef Name="No"/> </Key>  <Property Name="No" Nullable="false" Type="Edm.String"/>  <Property Name="Description"
Nullable="true" Type="Edm.String"/>  <Property Name="Inactive"
Nullable="false" Type="Edm.Boolean"/>  <Property Name="Insured"
Nullable="false" Type="Edm.Boolean"/> </EntityType> 

<EntityType Name="PAssetsDepreciationBook">

<EntityType Name="PBankAPostGrp">
<EntityType Name="PCompany">
<EntityType Name="PCustomer">

是否可以在C#aplication中获取此$metadata中的信息?

我有一个应用程序,其中包含对服务工作的引用,以及我正在使用的一些代码:

  

uriString = String.Format(“PAssetsDepreciationBook?$ filter = FA_No eq   '{0}”',cliente);

     

contex = new ServiceReference1.NAV(new Uri(serviceEndPoint));   contex.Credentials = CredentialCache.DefaultCredentials;

     

var customers = contex.Execute(new Uri(uriString,UriKind.Relative));

     

foreach(客户中的var c)   {      result = result + c.Acquisition_Cost;   }   返回结果;

这种方法很好,但要$metadata没有。

3 个答案:

答案 0 :(得分:3)

已经有代码(大部分)来执行此操作。它涉及 ODataMessageReader.ReadMetadataDocument()调用:

var request = WebRequest.CreateHttp(baseUrl + "$metadata");
var metadataMessage =
    new ClientHttpResponseMessage((HttpWebResponse)request.GetResponse());
using (var messageReader = new ODataMessageReader(metadataMessage))
{
    IEdmModel edmModel = messageReader.ReadMetadataDocument();
    // Do stuff with edmModel here
}

您需要ClientHttpResponseMessage类,但这很简单(来自ODataLib101):

public class ClientHttpResponseMessage : IODataResponseMessage
{
    private readonly HttpWebResponse webResponse;

    public ClientHttpResponseMessage(HttpWebResponse webResponse)
    {
        if (webResponse == null)
            throw new ArgumentNullException("webResponse");
        this.webResponse = webResponse;
    }

    public IEnumerable<KeyValuePair<string, string>> Headers
    {
        get
        {
            return this.webResponse.Headers.AllKeys.Select(
                headerName => new KeyValuePair<string, string>(
                   headerName, this.webResponse.Headers.Get(headerName)));
        }
    }

    public int StatusCode
    {
        get { return (int)this.webResponse.StatusCode; }
        set
        {
            throw new InvalidOperationException(
                "The HTTP response is read-only, status code can't be modified on it.");
        }
    }

    public Stream GetStream()
    {
        return this.webResponse.GetResponseStream();
    }

    public string GetHeader(string headerName)
    {
        if (headerName == null)
            throw new ArgumentNullException("headerName");
        return this.webResponse.Headers.Get(headerName);
    }

    public void SetHeader(string headerName, string headerValue)
    {
        throw new InvalidOperationException(
            "The HTTP response is read-only, headers can't be modified on it.");
    }
}

使用IEdmModel,您可以访问元数据中的所有信息来执行诸如构建类型 - >控制器名称的字典之类的内容:

Dictionary<type, string> typeControllerMap =
    edmModel.SchemaElements.OfType<IEdmEntityContainer>()
            .Single()
            .Elements.OfType<IEdmEntitySet>()
            .Select(es => new { t = FindType(es.ElementType.FullName()), n = es.Name })
            .Where(tn => tn.t != null)
            .ToDictionary(tn => tn.t, tn => tn.n);

上面LINQ链中间的 FindType()调用只是在给定类型名称的类型中搜索所有程序集:

private static Type FindType(string fullName)
{
    return
        AppDomain.CurrentDomain.GetAssemblies()
            .Where(a => !a.IsDynamic)
            .SelectMany(a => a.GetTypes())
            .FirstOrDefault(t => t.FullName.Equals(fullName));
}

答案 1 :(得分:1)

例如,您可以使用HttpWebRequest以纯XML格式请求元数据。如果您需要解析它,可以使用EdmLibMicrosoft.Data.Edm.dll),NuGet或更好ODataLibMicrosoft.Data.OData.dll){ {1}} NugetODataMessageReader专为阅读这些而设计(它仍为其返回EDM对象模型,它只是为您处理ReadMetadataDocument包装和版本控制)。

答案 2 :(得分:1)

谢谢,它运作得很好。

就像这样:

// Create a request for the URL. 
WebRequest request = WebRequest.Create("http://localhost:7048/DynamicsNAV70/OData/$metadata");

//set the credentials.
request.Credentials = CredentialCache.DefaultCredentials;

// Get the response.
WebResponse response = request.GetResponse();

// Get the stream containing content returned by the server.
Stream dataStream = response.GetResponseStream();

// Open the stream using a StreamReader for easy access.
StreamReader reader = new StreamReader(dataStream);

// Read the content.
string responseFromServer = reader.ReadToEnd();

XmlDocument xmlDoc = new XmlDocument();
xmlDoc.Load(new StringReader(responseFromServer));

// Clean up the streams and the response.
reader.Close();
response.Close();

在此之后,只需像xml文件一样解析数据。