Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

C# - 인터넷 시간 서버로부터 받은 시간을 윈도우에 적용하는 방법

다음과 같은 질문이 있군요. ^^

인터넷 시간을 불러와 pc에 적용 시키고 싶습니다.
; https://www.sysnet.pe.kr/3/0/5152

코드를 보면, 공개된 time.nist.gov 시간 서버(NTP: Network Time Protocol)로부터 데이터를 받아와 처리하고 있습니다.

/*
c:\temp> telnet time.nist.gov 13
59442 21-08-16 14:28:19 50 0 0 585.3 UTC(NIST) *

Connection to host lost.
*/
using System;
using System.Diagnostics;
using System.Globalization;
using System.IO;
using System.Net.Sockets;
using System.Runtime.InteropServices;
using System.Threading;

namespace ConsoleApp_test
{
    class Program
    {
        [StructLayout(LayoutKind.Sequential)]
        public struct SYSTEMTIME
        {
            public short wYear;
            public short wMonth;
            public short wDayOfWeek;
            public short wDay;
            public short wHour;
            public short wMinute;
            public short wSecond;
            public short wMilliseconds;
        }

        [DllImport("kernel32.dll", SetLastError = true)]
        public static extern bool SetSystemTime(ref SYSTEMTIME st);

        [DllImport("kernel32.dll", SetLastError = true)]
        public static extern bool SetLocalTime(ref SYSTEMTIME st);

        static void Main(string[] args)
        {
            Console.WriteLine(Process.GetCurrentProcess().Id);
            while (true)
            {
                string responseText = null;

                try
                {
                    using (var client = new TcpClient("time.nist.gov", 13))
                    using (var streamReader = new StreamReader(client.GetStream()))
                    {
                        Thread.Sleep(5000);

                        // 인터넷 시간 불러오기
                        responseText = streamReader.ReadToEnd(); // "59442 21-08-16 14:28:19 50 0 0 585.3 UTC(NIST) *"
                        var utcDateTimeString = responseText.Substring(7, 17);

                        if (DateTime.TryParseExact(utcDateTimeString, "yy-MM-dd HH:mm:ss", CultureInfo.InvariantCulture, DateTimeStyles.AssumeUniversal, out DateTime utcDateTime) == false)
                        {
                            Console.WriteLine(responseText);
                            continue;
                        }

                        var localDateTime = utcDateTime.ToLocalTime();

                        // pc에 적용하기
                        SYSTEMTIME st = new SYSTEMTIME();
                        st.wYear = (short)localDateTime.Year;
                        st.wMonth = (short)localDateTime.Month;
                        st.wDay = (short)localDateTime.Day;
                        st.wHour = (short)localDateTime.Hour;
                        st.wMinute = (short)localDateTime.Minute;
                        st.wSecond = (short)localDateTime.Second;
                        st.wMilliseconds = (short)localDateTime.Millisecond;

                        bool result = SetLocalTime(ref st);
                        if (result == false)
                        {
                            int lastError = Marshal.GetLastWin32Error();
                            Console.WriteLine(lastError);
                        }

                        Console.WriteLine($"Response: {responseText}, DateTime: {localDateTime}");
                    }
                }
                catch (Exception e)
                {
                    Console.WriteLine("Exception: " + e.Message + "\n(" + responseText + ")");
                }
            }
        }
    }
}

하지만, time.nist.gov로부터 시간은 잘 받아왔는데 윈도우 운영체제의 시간을 설정하는 SetSystemTime/SetLocalTime Win32 API 호출 시 결과가 false를 반환합니다. 이런 경우 Marshal.GetLastWin32Error 메서드를 이용하면 Win32 API가 왜 실패했는지를 알 수 있는데, 위의 경우에는 1314 값으로 이것을 Visual Studio의 "Tools" / "Error Lookup" 도구에서 살펴보면,

errlookup_1.png

"A required privilege is not held by the client."라는 설명이 나옵니다. 역시 왜 이런 식의 오류가 발생했는지는 문서를 보면 됩니다.

The calling process must have the SE_SYSTEMTIME_NAME privilege. This privilege is disabled by default. The SetSystemTime function enables the SE_SYSTEMTIME_NAME privilege before changing the system time and disables the privilege before returning.


그렇습니다. 운영체제의 시간을 변경하려면 호출 프로세스의 권한에 SE_SYSTEMTIME_NAME 특권이 있어야 하는데 윈도우의 일반 사용자 권한에는 그 특권이 없기 때문에 1314 오류가 발생하는 것입니다. 따라서 이 프로그램을 관리자 권한으로 실행하면 문제는 간단하게 해결됩니다.

참고로, 특권에 대해서는 예전에 쓴 글이 있습니다.

SeCreateGlobalPrivilege 특권과 WCF NamedPipe
; https://www.sysnet.pe.kr/2/0/1806

일반적으로 관리자 권한의 경우 다행히 SE_SYSTEMTIME_NAME 특권이 기본적으로 활성화된 상태이므로 일부러 설정할 필요까지는 없습니다. 만약 관리자 권한에서도 SE_SYSTEMTIME_NAME 특권이 없었다고 가정한다면, 다음의 NuGet 라이브러리를 이용해,

Win32.TokenPrivileges
; https://github.com/trondr/Win32.TokenPrivileges

다음과 같은 식으로 강제 활성화 후 SetSystemTime/SetLocalTime API를 호출하면 됩니다.

using (new AdjustPrivilege(PrivilegeName.SeSystemtimePrivilege))
{
    Console.WriteLine("Privileges should now be granted.");
    Console.WriteLine($"SeSystemtimePrivilege: {PrivilegeProvider.HasPrivilege(null, currentProcess, PrivilegeName.SeSystemtimePrivilege)}");

    bool result = SetLocalTime(ref st);
    if (result == false)
    {
        int lastError = Marshal.GetLastWin32Error();
        Console.WriteLine(lastError);
    }
}

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/10/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 31  [32]  33  34  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12831정성태9/6/20216328VC++: 148. Golang - 채널에 따른 다중 작업 처리파일 다운로드1
12830정성태9/6/20218609오류 유형: 761. Internet Explorer에서 파일 다운로드 시 "Your current security settings do not allow this file to be downloaded." 오류
12829정성태9/5/202110217.NET Framework: 1115. C# 10 - (14) 구조체 타입에 기본 생성자 정의 가능파일 다운로드1
12828정성태9/4/20218369.NET Framework: 1114. C# 10 - (13) 단일 파일 내에 적용되는 namespace 선언파일 다운로드1
12827정성태9/4/20218322스크립트: 27. 파이썬 - 웹 페이지 데이터 수집을 위한 scrapy Crawler 사용법 요약
12826정성태9/3/202110538.NET Framework: 1113. C# 10 - (12) 문자열 보간 성능 개선 [1]파일 다운로드1
12825정성태9/3/20218119개발 환경 구성: 603. GoLand - WSL 환경과 연동
12824정성태9/2/202117205오류 유형: 760. 파이썬 tensorflow - Dst tensor is not initialized. 오류 메시지
12823정성태9/2/20216870스크립트: 26. 파이썬 - PyCharm을 이용한 fork 디버그 방법
12822정성태9/1/202112100오류 유형: 759. 파이썬 tensorflow - ValueError: Shapes (...) and (...) are incompatible [2]
12821정성태9/1/20217697.NET Framework: 1112. C# - .NET 6부터 공개된 ISpanFormattable 사용법
12820정성태9/1/20218009VC++: 147. Golang - try/catch에 대응하는 panic/recover [1]파일 다운로드1
12819정성태8/31/20218101.NET Framework: 1111. C# - FormattableString 타입
12818정성태8/31/20217335Windows: 198. 윈도우 - 작업 관리자에서 (tensorflow 등으로 인한) GPU 연산 부하 보는 방법
12817정성태8/31/20219887스크립트: 25. 파이썬 - 윈도우 환경에서 directml을 이용한 tensorflow의 AMD GPU 사용 방법
12816정성태8/30/202115261스크립트: 24. 파이썬 - tensorflow 2.6 NVidia GPU 사용 방법 [2]
12815정성태8/30/20218387개발 환경 구성: 602. WSL 2 - docker-desktop-data, docker-desktop (%LOCALAPPDATA%\Docker\wsl\data\ext4.vhdx) 파일을 다른 디렉터리로 옮기는 방법
12814정성태8/30/202110700.NET Framework: 1110. C# 11 - 인터페이스 내에 정적 추상 메서드 정의 가능 (DIM for Static Members) [2]파일 다운로드1
12813정성태8/29/20218908.NET Framework: 1109. C# 10 - (11) Lambda 개선파일 다운로드1
12812정성태8/28/20218550.NET Framework: 1108. C# 10 - (10) 개선된 #line 지시자
12811정성태8/27/20218781Linux: 44. 윈도우 개발자를 위한 리눅스 fork 동작 방식 설명 (파이썬 코드)
12810정성태8/27/20217526.NET Framework: 1107. .NET Core/5+에서 동적 컴파일한 C# 코드를 (Breakpoint도 활용하며) 디버깅하는 방법 - #line 지시자파일 다운로드1
12809정성태8/26/20218205.NET Framework: 1106. .NET Core/5+에서 C# 코드를 동적으로 컴파일/사용하는 방법 [1]파일 다운로드1
12808정성태8/25/20219445오류 유형: 758. go: ...: missing go.sum entry; to add it: go mod download ...
12807정성태8/25/20219408.NET Framework: 1105. C# 10 - (9) 비동기 메서드가 사용할 AsyncMethodBuilder 선택 가능파일 다운로드1
12806정성태8/24/20217004개발 환경 구성: 601. PyCharm - 다중 프로세스 디버깅 방법
... 31  [32]  33  34  35  36  37  38  39  40  41  42  43  44  45  ...