성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] VT sequences to "CONOUT$" vs. STD_O...
[정성태] NetCoreDbg is a managed code debugg...
[정성태] Evaluating tail call elimination in...
[정성태] What’s new in System.Text.Json in ....
[정성태] What's new in .NET 9: Cryptography ...
[정성태] 아... 제시해 주신 "https://akrzemi1.wordp...
[정성태] 다시 질문을 정리할 필요가 있을 것 같습니다. 제가 본문에...
[이승준] 완전히 잘못 짚었습니다. 댓글 지우고 싶네요. 검색을 해보...
[정성태] 우선 답글 감사합니다. ^^ 그런데, 사실 저 예제는 (g...
[이승준] 수정이 안되어서... byteArray는 BYTE* 타입입니다...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>C# - 인터넷 시간 서버로부터 받은 시간을 윈도우에 적용하는 방법</h1> <p> 다음과 같은 질문이 있군요. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 인터넷 시간을 불러와 pc에 적용 시키고 싶습니다. ; <a target='tab' href='http://www.sysnet.pe.kr/3/0/5152'>http://www.sysnet.pe.kr/3/0/5152</a> </pre> <br /> 코드를 보면, 공개된 time.nist.gov <a target='tab' href='https://www.codeproject.com/Tips/5367263/NTP-SNTP-v4-v3-Server-Client-Cplusplus'>시간 서버</a>(NTP: Network Time Protocol)로부터 데이터를 받아와 처리하고 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > /* 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); <span style='color: blue; font-weight: bold'>[DllImport("kernel32.dll", SetLastError = true)] public static extern bool SetLocalTime(ref SYSTEMTIME st);</span> 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; } <span style='color: blue; font-weight: bold'>var localDateTime = utcDateTime.ToLocalTime();</span> // 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; <span style='color: blue; font-weight: bold'>bool result = SetLocalTime(ref st); if (result == false) { int lastError = Marshal.GetLastWin32Error(); Console.WriteLine(lastError); }</span> Console.WriteLine($"Response: {responseText}, DateTime: {localDateTime}"); } } catch (Exception e) { Console.WriteLine("Exception: " + e.Message + "\n(" + responseText + ")"); } } } } } </pre> <br /> 하지만, time.nist.gov로부터 시간은 잘 받아왔는데 윈도우 운영체제의 시간을 설정하는 <a target='tab' href='https://learn.microsoft.com/en-us/windows/desktop/api/sysinfoapi/nf-sysinfoapi-setsystemtime'>SetSystemTime/SetLocalTime Win32 API</a> 호출 시 결과가 false를 반환합니다. 이런 경우 Marshal.GetLastWin32Error 메서드를 이용하면 Win32 API가 왜 실패했는지를 알 수 있는데, 위의 경우에는 1314 값으로 이것을 Visual Studio의 "Tools" / "Error Lookup" 도구에서 살펴보면,<br /> <br /> <img alt='errlookup_1.png' src='/SysWebRes/bbs/errlookup_1.png' /><br /> <br /> "A required privilege is not held by the client."라는 설명이 나옵니다. 역시 왜 이런 식의 오류가 발생했는지는 문서를 보면 됩니다.<br /> <br /> <div style='BACKGROUND-COLOR: #ccffcc; padding: 10px 10px 5px 10px; MARGIN: 0px 10px 10px 10px; FONT-FAMILY: Malgun Gothic, Consolas, Verdana; COLOR: #005555'> <span style='color: blue; font-weight: bold'>The calling process must have the SE_SYSTEMTIME_NAME privilege</span>. 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. </div><br /> <br /> 그렇습니다. 운영체제의 시간을 변경하려면 호출 프로세스의 권한에 SE_SYSTEMTIME_NAME 특권이 있어야 하는데 윈도우의 일반 사용자 권한에는 그 특권이 없기 때문에 1314 오류가 발생하는 것입니다. 따라서 이 프로그램을 관리자 권한으로 실행하면 문제는 간단하게 해결됩니다.<br /> <br /> 참고로, 특권에 대해서는 예전에 쓴 글이 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > SeCreateGlobalPrivilege 특권과 WCF NamedPipe ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/1806'>http://www.sysnet.pe.kr/2/0/1806</a> </pre> <br /> 일반적으로 관리자 권한의 경우 다행히 SE_SYSTEMTIME_NAME 특권이 기본적으로 활성화된 상태이므로 일부러 설정할 필요까지는 없습니다. 만약 관리자 권한에서도 SE_SYSTEMTIME_NAME 특권이 없었다고 가정한다면, 다음의 NuGet 라이브러리를 이용해,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Win32.TokenPrivileges ; <a target='tab' href='https://github.com/trondr/Win32.TokenPrivileges'>https://github.com/trondr/Win32.TokenPrivileges</a> </pre> <br /> 다음과 같은 식으로 강제 활성화 후 SetSystemTime/SetLocalTime API를 호출하면 됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 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); } } </pre> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1447&boardid=331301885'>첨부 파일은 이 글의 예제 코드를 포함</a>합니다.)<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1591
(왼쪽의 숫자를 입력해야 합니다.)