[키보드] Fujitsu – Libertouch

후지츠 리버터치

  • 장점: 부드러운 키감.
  • 단점: 87키가 아닌점.

후지츠 리버터치(출처: 아마존재팬)

 

 

일본 IT 전시회를 관람하기 위해 방문 한 김에 한국에서는 판매하지 않는 리버터치 를 구매했습니다. 리버터치는 중고나라에서 봐도 아직도 찾는 사람이 있을 정도로 키보드에 대한 평이 좋습니다.

한가지 키보드를 너무 오래 사용한 느낌이 있어 출장비를 아껴 키보드를 구매했습니다. 원래는 87키를 구매하려고 했으나 제가 착오를 한건지 오프라인 매장에서 판매를 안하는 건지 요도바시 카메라 아키하바라 점에서 찾질 못하고 어쩔수 없이 103키를 구매했습니다.

출장후에 쭉 사용하고 있는데 느낌이 아주 부드러운게 미끄러지는 듯한 타이핑이 가능합니다.  이 키보드를 쓰다가 다른 키보드를 못쓸정도로요..^^

작업환경이 여러개라 Mac에는 해피해킹 Windows에는 Libertouch 를 연결해서 사용하는데 해피해킹 으로 가서 타이핑을 하면 묵찍하다는 느낌이 듭니다. 그만큼 Libertouch 가 부드럽다는 것이겠죠..

[키보드] Truly Ergonomic

키보드 후기


 

  • 장점:  타이핑이 정말 편안하다.
  • 단점
    • 키 배열에 익숙해지는데 시간이 많이 걸린다. (처음치는 순간 멘붕)
    • 조합이 필요한 키를 사용하기가 어려울때가 있다.
    • Tab 키가 가운데 있어 shell 에서 자동완성 기능을 사용할 때 불편하다.

The only Truly Ergonomic Keyboard with mechanical keyswitches

2년전쯤에 키보드 타이핑중에 손목 통증이 있어 인체공학 디자인의 키보드를 찾고 있었습니다. 인체공학 키보드는 대학교때 사용하던 MS Natural keyboard 를 마지막으로 써보고 다른 제품은 아직 못써보고 있었습니다.
그렇지만 중고나라 눈팅중에 Truly Ergonomic 키보드를 발견하고는 직거래로 바로 받아왔습니다.

6개월 정도 사용했는데 단점도 많지만 손목 통증은 조금 완화되는 느낌입니다. 정확하게 제가 병원에서 진단을 받은 것도 아니기 때문에 어느정도 좋다라는 객관적인 증거는 없습니다. ^^

지금은 키보드 스위치가 몇개 고장나서 수리를 해야해서 못쓰고 있지만 조만간 사용할 예정입니다.

[캠핑후기] 포천 낭만가족캠핑장

포천 낭만 가족 캠핑장


  • 일정: 2016. 06. 04~2016. 06. 06 (2박 3일)
  • 누구랑: 안지기 아는 사람 가족과
  • 장점: 수영장, 인공 계곡(?)
  • 단점: 화장실과의 거리, 진입로 및 모래언덕

지난 주말 첫째가 캠핑, 캠핑 노래를 불러 다른 가족과 함께 캠핑을 다녀왔습니다.
같은날 캠핑하는 분들이 모두 매너가 좋은 분들만 모여 있었는지 아주 조용하고 편한하게 캠핑을 하고 왔습니다.

오랜만에 별도 많이 보고 나무도 많이 보고 했더니 힐링 되는 듯한 느낌이었습니다. 아이들과 너무 열심히 놀아서인지 몰라도 팔에 근육통도 생기고…^^

한가지 아쉬웠던 점은 진입로 및 캠핑장 내부 도로가 모래로 되어 있고 파인곳도 많아서 아이들이 미끄러지거나 차가 미끄러져서 휠스핀을 하는 것을 많이 봤습니다. 차는 상관 없었지만 아이들이 위험한게 조금 아쉬웠습니다.

그래도 최근에 일에 치여 살다가 나가서 콧바람을 쐬니 아주 좋았습니다.^^

[Git] 문서파일 비교

git 레포지토리에는 MS-OFFICE 문서 파일이 포함되는 경우가 많이 있습니다.

만약 본인 또는 다른 사람이 문서 파일을 변경한 경우 문서의 어떤 부분이 변경 되었는지 확인 할 수 있다면 보다 효율적으로 업무를 볼 수 있습니다.

해당 내용은 Pro Git 이라는 책에도 어느정도 설명이 된 부분이지만 실제로 테스트를 해가며 적용한 내용을 공유하고자 합니다.

DOC[X] 비교

문서 파일을 비교하는 것은 pandocApache Tika 를 사용하는 방법이 있습니다. 물론 이외에 여러가지 방법이 있습니다.

 

pandoc 적용방법

1. Git Config 파일 수정

아래의 내용을 `.gitconfig` 파일에 추가합니다. `USER_HOME` 디렉토리는 `C:/Users/로그인계정` 입니다.

[USER_HOME/.gitconfig]

[diff "pandoc"]
textconv=pandoc --to=markdown
prompt = false

[alias]
wdiff = diff --word-diff=color --unified=1

2. Git Repository 디렉토리에 .gitattribute 파일 추가

Git 레포지토리로 이동하여 .gitattribute 파일을 생성한 후 아래의 내용을 넣습니다.

*.docx diff=pandoc

3. 문서 변경 부분 확인

저는 임의로 `test.docx` 라는 문서를 만들고 테스트를 했습니다.
`test.docx` 라는 파일에는 애국가가 적혀 있습니다.

[test.docx]

동해물과 백수산이 마르고 닳도록 하느님이 보우하사 우리나라만세.

이후 무궁화 삼천리 화려강산 대한 사람 대한으로 길이 보전하세 라는 후렴부를 추가했습니다.
비교된 결과를 확인해보겠습니다.

$ git diff
diff --git a/test.docx b/test.docx
index 44c02f2..27dc279 100644
--- a/test.docx
+++ b/test.docx
@@ -1 +1,3 @@
동해물과 백수산이 마르고 닳도록 하느님이 보우하사 우리나라만세.
+^M
+무궁화 삼천리 화려강산 대한 사람 대한으로 우리나라 만세.^M

불필요한 글자가 출력되기는 하지만 수정된 내용이 표시되었습니다.
무궁화 삼천리 화려강산 대한 사람 대한으로 길이 보전하세 소절에서 화려강산 이란 단어를 삭제하였습니다.

$ git diff
diff --git a/test.docx b/test.docx
index 27dc279..71bbf31 100644
--- a/test.docx
+++ b/test.docx
@@ -1,3 +1,3 @@
동해물과 백수산이 마르고 닳도록 하느님이 보우하사 우리나라만세.

-무궁화 삼천리 화려강산 대한 사람 대한으로 길이 보전하세.
+무궁화 삼천리 대한 사람 대한으로 길이 보전하세.^M

변경된 라인이 모두 출력됩니다.


Tika 적용방법

Apache Tika 의 경우 PPTX 포맷과 DOCX 포맷을 모두 지원한다는 장점이 있습니다.

1. Tika 다운로드

https://tika.apache.org/download.html 에서 tika-app 항목을 다운로드 합니다. 제가 테스트 하는 시점에서 tika 버전은 1.12 입니다.

2. 환경 변수 설정

다운로드한 tika-app-1.12.jar 파일을 환경 변수에 추가하거나 저장 위치를 확인합니다.

3. 배치 파일 생성

배치 파일은 .gitconfig 에서 호출하기 위함과 파라미터를 받기 위함입니다. tika 를 실행할 때 사용하는 옵션에는 여러종류가 있습니다. (링크참조)

[USER_HOME/tika.bat]

java -jar [TIKA 저장경로] -t --encoding=UTF-8 %1

4. `.gitconfig` 파일 수정

위에서 `pandoc` 적용 방법에서 수정한 `.gitconfig` 파일을 다시 수정합니다.
아래 부분에서 `openxml` 부분이 추가 되었습니다. `textconv` 속성에 위에서 작성한 `tika.bat` 파일의 경로를 지정합니다.

[USER_HOME/.gitconfig]

[diff "pandoc"]
textconv=pandoc --to=markdown
prompt = false
[alias]
wdiff = diff --word-diff=color --unified=1
[diff "openxml"]
textconv=C:/Users/coozplz/tika.bat

5. Git Repository 디렉토리에 .gitattribute 파일 수정

위에서 설정된 .gitattribute 파일을 아래와 같이 수정합니다.

*.docx diff=openxml
*.pptx diff=openxml

6. 문서 변경 부분 확인

위에서 수정한 내용을 커밋한 상태가 아니기 때문에 동일한 내용이 다시 출력됩니다.

$ git diff
diff --git a/.gitattributes b/.gitattributes
index 51cead2..ef6a520 100644
--- a/.gitattributes
+++ b/.gitattributes
@@ -1,2 +1,2 @@
-*.docx diff=pandoc
+*.docx diff=openxml
*.pptx diff=openxml
warning: LF will be replaced by CRLF in .gitattributes.
The file will have its original line endings in your working directory.
diff --git a/test.docx b/test.docx
index 27dc279..71bbf31 100644
--- a/test.docx
+++ b/test.docx
@@ -1,4 +1,4 @@

-D:\temp>java -jar "C:\\Users\\coozplz\\tika.jar" -t --encoding=UTF-8 C:/Users/coozplz/AppData/Local/Temp/IjVuPa_test.docx
+D:\temp>java -jar "C:\\Users\\coozplz\\tika.jar" -t --encoding=UTF-8 test.docx ^M
동해물과 백수산이 마르고 닳도록 하느님이 보우하사 우리나라만세.
-무궁화 삼천리 화려강산 대한 사람 대한으로 길이보전하세.
+무궁화 삼천리 대한 사람 대한으로 길이보전하세

실행이 약간 느리기는 하나 문제없이 출력되는 것을 확인할 수 있습니다.

PPTX 변경 내용 확인

test.docx 문서와 마찬가지로 애국가 1절을 후렴부 없이 입력하고 커밋한 후 후렴부를 추가하였습니다.

$ git diff
diff --git a/test.pptx b/test.pptx
index 64fc17b..09ecd25 100644
--- a/test.pptx
+++ b/test.pptx
@@ -1,9 +1,10 @@

-D:\temp>java -jar "c:\\Users\\coozplz\\tika.jar" -t --encoding=UTF-8 C:/Users/coozplz/AppData/Local/Temp/F4N97a_test.pptx
+D:\temp>java -jar "c:\\Users\\coozplz\\tika.jar" -t --encoding=UTF-8 test.pptx ^M
애국가
1절
동해물과 백두산이 마르고 닳도록 하느님이 보우하사 우리나라 만세.
+무궁화 삼천리 화려강산 대한사람 대한으로 길이 보전하세.

PPT문서를 텍스트 형태로 비교하는 내용을 확인 할 수 있습니다.

위에서 언급은 하지 않았지만 xlsx 문서 및 이미지 파일 비교도 가능합니다.

해당 내용은 나중에 올리도록 하겠습니다.

이 글은 Tools 카테고리에 분류되었고 태그가 있으며 님에 의해 에 작성되었습니다.

[Shell Script] 인라인 파이썬코드로 ini 파일 파싱

Shell Script에서 python 을 이용하여 ini 파일 파싱


의도

쉘 스크립트를 이용해 /etc/systemd/system/서비스명 파일을 분석하려고 하니 스크립트에서 ini 형태의 파일을 분석하는게 조금 노가다성이라 pythonConfigParser 를 이용하면 좋겠다라는 생각이 들어 사용하였습니다.

[/etc/systemd/system/web-monitor.service] 샘플

[Unit]
Description=CHIeru Webmonitoring system.

[Service]
ExecStart=/usr/bin/node $INSTALL_PATH/server/server.js
Restart=always
StandardOutput=syslog
StandardError=syslog
SyslogIdentifier=web-monitor
User=root
Environment=\"NODE_CONFIG_DIR=$INSTALL_PATH/config\"

[Install]
WantedBy=multi-user.target

위의 부분에서 제가 필요한 부분은 Service 섹션의 ExecStart 값중 INSTALL_PATH 경로를 가져오는 것입니다.

[Shell Script code]

pathFromService=$(python - <<EOF
import ConfigParser
import os
Config = ConfigParser.ConfigParser()
Config.read('/etc/systemd/system/web-monitor.service')
executeCommand = Config.get('Service', 'ExecStart')
executeFilePath = executeCommand.split(' ')[1]

installationPath = os.path.abspath(os.path.dirname(executeFilePath) +'/../')

print installationPath
EOF
)
    LEN=$(echo ${#pathFromService})
    echo $pathFromService

[AutoHotKey] 볼륨 제어 스크립트

키보드로 리얼포스(REALFORCE)를 사용하면서 불편함이 미디어를 제어할 수 있는 키가 없다는 점이 불편했습니다.

사무실에서 음악을 듣다가 주변 사람이 말을 걸면 음악을 멈춰야 되는데 매번 마우스로 클릭해서 음소거 기능을 켜자니 불편해서 찾아보다보니 AutoHotKey를 이용해서 볼륨제어가 가능하다는 것을 알게 되었습니다.

키보드 매핑 프로그램을 여러가지가 있지만 간단하게 제가 원하는 기능만 사용하면 되므로 매핑 프로그램을 사용할 필요는 없었습니다.

아래 스크립트를 컴파일 해서 시작 프로그램에 넣으면 끝납니다.

[Script]

;------------------------------------------------------------------------------
; File : volumn_controls.ahk
; Author: coozplz@gmail.com
; Date: 2016. 04. 06
; Desc: 볼륨 컨트롤 키가 없는 키보드에서 음량을 제어 하는 기능

;------------------------------------------------------------------------------
;                               단축키 목록
;------------------------------------------------------------------------------
;                   [Left or Right] Window + PageUp   : 음량 크게
;                   [Left or Right] Window + PageDown : 음량 작게
;                   [Left or Right] Window + End      : 음소거
;------------------------------------------------------------------------------

#NoEnv  ; Recommended for performance and compatibility with future AutoHotkey releases.
; #Warn  ; Enable warnings to assist with detecting common errors.
SendMode Input  ; Recommended for new scripts due to its superior speed and reliability.
SetWorkingDir %A_ScriptDir%  ; Ensures a consistent starting directory.

; Volume Up settings.
<#>PgUp::SoundSet +10
Return

; Volume Down setting.
<#>PgDn::SoundSet -10
Return

; Mute toggle settings.
<#>End::SoundSet, +1, , mute
Return

[C] Echo server/client

대학교 이후로 거의 접해지 못했던 C 를 다시 시작하고 있습니다.

다시 시작하는 이유는 회사에서 C 개발자가 필요하기 때문입니다. 처음이라 진입이 쉽지는 않겠지만 포기하지 않고 계속 하다 보면 언젠가는 되리라 생각합니다.

예제는 http://www.joinc.co.kr/w/Site/win_network_prog/doc/winsock_basic 의 코드를 참조하며 모르는 부분에 주석을 추가하였습니다.

지속적으로 이전에 java로 만들었던 채팅 서버, 메신저 서버 등을 만들어볼 생각입니다.

개발환경

  • IDE: Visual Studio 2012
  • OS: Windows 10

[echo_server.c]

/*
File: echo_server.c
Desc: 에코 서버.
        서버는 클라이언트의 접속 요청을 수락한 후 송신된 메시지를 출력 및 동일 메시지 송신 후 클라이언트의 연결을 종료하는 프로그램
Date: 2016. 04. 27
Author: coozplz@gmail.com
*/
#include &lt;winsock.h&gt;
#include &lt;stdio.h&gt;

// Winsock 라이브러리 로딩을 위한 코드
#pragma comment(lib, &quot;Ws2_32.lib&quot;);


#define MAX_PACKET_LEN 512
#define PORT 5552
#define BACK_LOG_SIZE 5

int main()
{
    WSADATA wsaData;
    int status;
    int socketLength;
    int readSize, writeSize;
    SOCKET endPointSocket, clientSocket;

    struct sockaddr_in SocketInfo, ClientSocketInfo;

    char buffer[MAX_PACKET_LEN];

    if (WSAStartup(MAKEWORD(2, 2), &amp;wsaData) != 0) {
        printf(&quot;WSAStartup error \n&quot;);
        return 0;
    }

    endPointSocket = socket(AF_INET, SOCK_STREAM, 0);

    if (endPointSocket == INVALID_SOCKET) {
        printf(&quot;endPointSocket is invalid \n&quot;);
        return 0;
    }

    printf(&quot;Server is running on port %d\n&quot;, PORT);

    // 구조체를 초기화한다.
    ZeroMemory(&amp;SocketInfo, sizeof(struct sockaddr_in));
    SocketInfo.sin_family = AF_INET;
    // htons(): Host 시스템에서 Network 로 short 형 데이터를 보낼 때 바이트 오더를 바꿔주는 함수
    SocketInfo.sin_port = htons(PORT); 

    // htonl(): long형 데이터의 바이트 오더를 바꿔주는 함수.
    // INADDR_ANY 는 OS에 설정되어 있는 아이피 어느것이나 바인딩 한다는 의미
    SocketInfo.sin_addr.S_un.S_addr = htonl(INADDR_ANY); 

    status = bind(endPointSocket, (struct sockaddr*)&amp;SocketInfo, sizeof(struct sockaddr_in));

    if (status == SOCKET_ERROR) {
        printf(&quot;failed to bind socket \n&quot;);
        return 0;
    }

    //
    // backlog 인자는 아직 미결인 연결들에 대한 큐의 늘어날 수 있는 최대 길이를 정의한다.
    // 큐에 도착한 연결 요청들이 꽉 찬다면 클라이언트는 ECONNREFUSED를 가리키는 에러를 받거나 만일 하위 프로토콜이 재전송을 지원한다면, 요청은 재시도가 성공되도록 하기 위해 무시된다.
    //
    status = listen(endPointSocket, BACK_LOG_SIZE);

    if (status == SOCKET_ERROR) {
        printf(&quot;failed to listen socket \n&quot;);
        return 0;
    }

    while (TRUE) {
        // 클라이언트 구조체 초기화
        ZeroMemory(&amp;ClientSocketInfo, sizeof(struct sockaddr_in));
        socketLength = sizeof(struct sockaddr_in);

        clientSocket = accept(endPointSocket, (struct sockaddr *) &amp;ClientSocketInfo, &amp;socketLength);
        if (clientSocket == INVALID_SOCKET) {
            printf(&quot;client socket is invalid\n&quot;);
            closesocket(clientSocket);
            continue;
        }

        // inet_ntoa() : 네트워크 바이트 순서의 32비트 값을 Dotted-Decimal notation 의 주소값으로 변환
        // 예 0x6601a8c0: 102.1.168.192
        printf(&quot;[Accepted] %s:%d\n&quot;, inet_ntoa(ClientSocketInfo.sin_addr), (int)ntohs(ClientSocketInfo.sin_port));
        ZeroMemory(&amp;buffer, sizeof(buffer));
        readSize = recv(clientSocket, (void *)buffer, MAX_PACKET_LEN, 0);

        if (readSize &gt; 0) {
            printf(&quot;Message from client &gt; %s\n&quot;, buffer);
            writeSize = send(clientSocket, buffer, readSize, 0);
        }
        else {
            printf(&quot;Error on reading\n&quot;);
        }
        closesocket(clientSocket);
        printf(&quot;[Closed] %s:%d\n&quot;, inet_ntoa(ClientSocketInfo.sin_addr), (int)ntohs(ClientSocketInfo.sin_port));
    }
    closesocket(endPointSocket);
    WSACleanup();
    return 0;
}

[echo_client.c]

/*
File: echo_client.c
Desc: 에코 클라이언트.
      서버에 접속 요청을 한 후 사용자가 입력한 메시지를 서버에 송신 후 응답 메시지를 출력한 후 소켓 연결이 종료되었는지 판단하고 소켓 연결을 종료한다.
Date: 2016. 04. 27
Author: coozplz@gmail.com
*/

#include &lt;stdio.h&gt;
#include &lt;WinSock2.h&gt;

// Winsock 라이브러리 로딩을 위한 코드
#pragma comment(lib, &quot;Ws2_32.lib&quot;);


#define PORT_NUM 5552
#define MAX_LEN 1024

int main(int argc, char **argv)
{
    SOCKET socketFd;
    WSADATA wsaData;
    struct sockaddr_in addr;

    char buffer[MAX_LEN];
    char receiveBuffer[MAX_LEN];

    int status = 0;
    int readSize = 0;
    status = WSAStartup(MAKEWORD(2, 2), &amp;wsaData);

    if (status != NO_ERROR) {
        printf(&quot;WSAStart has error\n&quot;);
        return 1;
    }

    socketFd = socket(AF_INET, SOCK_STREAM, 0);

    if (socketFd == INVALID_SOCKET) {
        printf(&quot;socketFd is invalid\n&quot;);
        return 1;
    }
    ZeroMemory(&amp;addr, sizeof(struct sockaddr_in));

    addr.sin_family = AF_INET;
    //
    // inet_addr() : Dotted-decimal 타입을 네트워크 32바이트 순서로 변경
    //
    addr.sin_addr.S_un.S_addr = inet_addr(&quot;192.168.10.52&quot;);
    addr.sin_port = htons(PORT_NUM);

    status = connect(socketFd, (struct sockaddr_in *)&amp;addr, sizeof(addr));

    if (status == SOCKET_ERROR) {
        printf(&quot;connection failed...\n&quot;);
        return 1;
    }

    while (TRUE) {
        printf(&quot;Connection is established...\n&quot;);
        printf(&quot;&gt; &quot;);
        fgets(buffer, MAX_LEN - 1, stdin);
        if (strncmp(buffer, &quot;quit\n&quot;, 5) == 0) {
            break;
        }

        printf(&quot;SEND: %s&quot;, buffer);
        send(socketFd, (void *)buffer, strlen(buffer), 0);
        ZeroMemory(&amp;receiveBuffer, sizeof(receiveBuffer));
        readSize = recv(socketFd, (void *)receiveBuffer, MAX_LEN, 0);
        printf(&quot;READ = %s\n&quot;, receiveBuffer);
        //
        // 소켓연결이 종료되었는지 판단하기 위해 다시 recv() 함수를 호출한다.
        //
        ZeroMemory(&amp;receiveBuffer, sizeof(receiveBuffer));
        readSize = recv(socketFd, (void *)receiveBuffer, MAX_LEN, 0);
        if (readSize &lt;= 0) {
            printf(&quot;socket is disconnected by server!!\n&quot;);
            break;
        }
    }
    closesocket(socketFd);
    WSACleanup();

    printf(&quot;Press any key to exit program.&quot;);
    getchar();
    return 0;
}
이 글은 C 카테고리에 분류되었고 , 태그가 있으며 님에 의해 에 작성되었습니다.