Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

c# server quickstart broken #230

Open
DGuhr opened this issue Mar 28, 2025 · 2 comments
Open

c# server quickstart broken #230

DGuhr opened this issue Mar 28, 2025 · 2 comments
Labels
bug Something isn't working

Comments

@DGuhr
Copy link

DGuhr commented Mar 28, 2025

Describe the bug
On this page, when you use the C# SDK, there are 2 Bugs resulting in always getting a "404" response.

First bug: The part where we extract the forecastURL and request that json is missing in the example.
Second bug: The double langitude/longitude values are only working in geos where it's really e.g. "95.789" and not "95,789" (as, for example, here in germany).
This leads to bad requests.

To Reproduce
Steps to reproduce the behavior:

  1. build the c# example locally
  2. try getting a weather forecast for any city. see the 404 error appearing.

Expected behavior
Quickstart works.

Logs
No logs, but here's an example of a working GetForecast method:

[McpServerTool, Description("Get weather forecast for a location.")]
    public static async Task<string> GetForecast(
        HttpClient client,
        [Description("Latitude of the location.")] double latitude,
        [Description("Longitude of the location.")] double longitude)
    {
        try
        {
            // Format the coordinates properly
            var formattedLat = latitude.ToString("F4", System.Globalization.CultureInfo.InvariantCulture);
            var formattedLon = longitude.ToString("F4", System.Globalization.CultureInfo.InvariantCulture);
            
            var pointsResponse = await client.GetAsync($"/points/{formattedLat},{formattedLon}");
            var pointsContent = await pointsResponse.Content.ReadAsStringAsync();
            
            if (!pointsResponse.IsSuccessStatusCode)
            {
                return $"Error retrieving points data: {(int)pointsResponse.StatusCode} {pointsResponse.StatusCode}";
            }
            
            // Parse the forecast URL from the points response
            var forecastUrl = JsonDocument.Parse(pointsContent).RootElement.GetProperty("properties").GetProperty("forecast").GetString();
            
            if (string.IsNullOrEmpty(forecastUrl))
            {
                return "Could not retrieve forecast URL from points data.";
            }
            
            // Get the forecast with the full URL from the response
            var forecastResponse = await client.GetAsync(forecastUrl);
            var forecastContent = await forecastResponse.Content.ReadAsStringAsync();
            
            if (!forecastResponse.IsSuccessStatusCode)
            {
                return $"Error retrieving forecast: {(int)forecastResponse.StatusCode} {forecastResponse.StatusCode}";
            }
            
            var forecastJson = JsonDocument.Parse(forecastContent).RootElement;
            var periods = forecastJson.GetProperty("properties").GetProperty("periods").EnumerateArray();

            return string.Join("\n---\n", periods.Select(period => $"""
                    {period.GetProperty("name").GetString()}
                    Temperature: {period.GetProperty("temperature").GetInt32()}°F
                    Wind: {period.GetProperty("windSpeed").GetString()} {period.GetProperty("windDirection").GetString()}
                    Forecast: {period.GetProperty("detailedForecast").GetString()}
                    """));
        }
        catch (Exception ex)
        {
            return $"Error retrieving forecast: {ex.Message}";
        }
    }

Additional context
Searched for some already open bug but couldn't find any. If there is one, and a fix already (repo just shows the python example), please close this. thanks.

@DGuhr
Copy link
Author

DGuhr commented Mar 28, 2025

Cleaned up worldwide-working minimal example:

[McpServerTool,
 Description(@"
      Get weather forecast for a location.
      Args:
      Latitude: Latitude of the location.
      Longitude: Longitude of the location.
")]
public static async Task<string> GetForecast(
    HttpClient client,
    [Description("Latitude of the location.")] double latitude,
    [Description("Longitude of the location.")] double longitude)
{
    var jsonElement = await client.GetFromJsonAsync<JsonElement>($"/points/{latitude.ToString(CultureInfo.InvariantCulture)},{longitude.ToString(CultureInfo.InvariantCulture)}");
    var forecastUrl = jsonElement.GetProperty("properties").GetProperty("forecast").GetString();

    jsonElement = await client.GetFromJsonAsync<JsonElement>(forecastUrl);
    var periods = jsonElement.GetProperty("properties").GetProperty("periods").EnumerateArray();

    return string.Join("\n---\n", periods.Select(period => $"""
                                                            {period.GetProperty("name").GetString()}
                                                            Temperature: {period.GetProperty("temperature").GetInt32()}°F
                                                            Wind: {period.GetProperty("windSpeed").GetString()} {period.GetProperty("windDirection").GetString()}
                                                            Forecast: {period.GetProperty("detailedForecast").GetString()}
                                                            """));
}

@mikekidder
Copy link

@DGuhr -- added your last update to the PR to use the Globalization.

Only change is switching to string.Format to try and reduce line lengths.
Let me know if I got it right! I did limited testing on it.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

No branches or pull requests

2 participants