С# Универсальное приложение для Windows Как получить время из Интернета

У меня возникли проблемы с Raspberry Pi 3, работающим под управлением ядра Windows 10 IoT, поскольку системные часы имеют тенденцию к значительному дрейфу с течением времени. Я решил сделать программу, которая устанавливала бы системные часы на правильное время, которое устройство получало бы из интернета. Я уже знаю, как установить системные часы, но проблема в том, как узнать точное время из Интернета при запуске универсального приложения Windows C# в ядре Windows 10 IoT?

Я сделал следующее, чтобы получить время с сервера времени Windows на x86 и x64:

public static DateTime GetNetworkTime()
{
    //default Windows time server
    const string ntpServer = "time.windows.com";

    // NTP message size - 16 bytes of the digest (RFC 2030)
    var ntpData = new byte[48];

    //Setting the Leap Indicator, Version Number and Mode values
    ntpData[0] = 0x1B; //LI = 0 (no warning), VN = 3 (IPv4 only), Mode = 3 (Client Mode)

    var addresses = Dns.GetHostEntry(ntpServer).AddressList;

    //The UDP port number assigned to NTP is 123
    var ipEndPoint = new IPEndPoint(addresses[0], 123);
    //NTP uses UDP
    var socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);

    socket.Connect(ipEndPoint);

   //Stops code hang if NTP is blocked
   socket.ReceiveTimeout = 3000;     

   socket.Send(ntpData);
   socket.Receive(ntpData);
   socket.Close();

   //Offset to get to the "Transmit Timestamp" field (time at which the reply 
  //departed the server for the client, in 64-bit timestamp format."
  const byte serverReplyTime = 40;

  //Get the seconds part
  ulong intPart = BitConverter.ToUInt32(ntpData, serverReplyTime);

  //Get the seconds fraction
  ulong fractPart = BitConverter.ToUInt32(ntpData, serverReplyTime + 4);

  //Convert From big-endian to little-endian
  intPart = SwapEndianness(intPart);
  fractPart = SwapEndianness(fractPart);

  var milliseconds = (intPart * 1000) + ((fractPart * 1000) / 0x100000000L);

  //**UTC** time
  var networkDateTime = (new DateTime(1900, 1, 1, 0, 0, 0, DateTimeKind.Utc)).AddMilliseconds((long)milliseconds);

  return networkDateTime.ToLocalTime();
}

// stackoverflow.com/a/3294698/162671
static uint SwapEndianness(ulong x)
{
    return (uint) (((x & 0x000000ff) << 24) +
               ((x & 0x0000ff00) << 8) +
               ((x & 0x00ff0000) >> 8) +
               ((x & 0xff000000) >> 24));
}

но он не работает для универсальных приложений Windows, так как Dns и другие связанные классы недоступны для использования.


person The amateur programmer    schedule 02.05.2016    source источник


Ответы (1)


Опция 1

Избегайте проблемы и не используйте время в Интернете. Вместо этого используйте интегральную схему часов реального времени, например DS1302.

Вариант 2

Используйте классы универсальных приложений Windows 10 для подключения по протоколу UDP. Они используют DatagramSocket. Вот примерный код, реализующий сокет дейтаграммы и получающий ответ от сервера времени.

private async void ConnectToTimeServer(object sender, RoutedEventArgs e)
{
    var socket = new DatagramSocket();
    socket.MessageReceived += SocketMessageReceived;
    await socket.ConnectAsync(new HostName("time.windows.com"), "123");

    using (var dataWriter = new DataWriter(socket.OutputStream))
    {
        var ntpData = new byte[48];
        ntpData[0] = 0x1B;
        dataWriter.WriteBytes(ntpData);
        await dataWriter.StoreAsync();
    }
}

Затем сообщение получило следующий код события:

private void SocketMessageReceived(DatagramSocket sender, DatagramSocketMessageReceivedEventArgs args)
{
    try
    {
        using (var reader = args.GetDataReader())
        {
            byte[] response = new byte[48];
            reader.ReadBytes(response);

            YourMethodToParseNetworkTime(response);
        }
    }
    catch (Exception ex)
    {
        // Handle exceptions
    }
}
person Steven Hook    schedule 03.05.2016
comment
Спасибо за этот ответ. Это помогает решить проблему. В интернете почему-то найти что-то на эту тему было сложно. - person The amateur programmer; 03.05.2016
comment
Исходя из своего опыта, я бы рекомендовал DS3231 вместо 1302. Я обнаружил, что он более точен со значительно меньшим дрейфом. - person ObjectType; 03.05.2016