PHP에서 데이터를 json 문자열로 변환할 때 json_encode(mixed $value) 함수를 사용하게 된다. 이 함수를 이용해 개체를 변환할 때에도 활용할 수 있다. 기본적으로는 클래스에서 public인 프로퍼티에 대해서만 json으로 반환한다. protected나 private, 또는 데이터를 가공해 json으로 반환해야 한다면 해당 클래스에서 JsonSerializable 인터페이스를 구성해 어떤 형태로 변환할 것인지 정의할 수 있다. 이 인터페이스는 PHP 5.4.0 이상, PHP 7 에서 지원하고 있다.

<?php
class Student {
    public $first_name;
    public $last_name;
    protected $school;

    public function __construct($first_name, $last_name, $school) {
        $this->first_name = $first_name;
        $this->last_name = $last_name;
        $this->school = $school;
    }
}

$haruair = new Student("Edward", "Kim", "WeirdSchool");
print $haruair;


// result :
// {
//   "first_name": "Edward",
//   "last_name": "Kim"
// }
?>

public 프로퍼티만 필요로 한 경우라면 별도의 인터페이스 구성 없이도 사용할 수 있다. 다만 대부분의 라이브러리에서 protected 또는 private으로 프로퍼티를 작성하고 __get(), __set() 매직 메소드를 구현해 사용하고 있고 또 권장하고 있기 때문에 그런 경우엔 다음과 같이 JsonSerializable 인터페이스를 활용할 수 있다.

<?php
class Student implements JsonSerializable {
    public $first_name;
    public $last_name;
    protected $school;

    public function __construct($first_name, $last_name, $school) {
        $this->first_name = $first_name;
        $this->last_name = $last_name;
        $this->school = $school;
    }

    public function jsonSerialize() {
        return [
            'full_name' => "{$this->first_name}, {$this->last_name}",
            'first_name' => $this->first_name,
            'last_name' => $this->last_name,
            'school' => $this->school,
        ];
    }
}

$haruair = new Student("Edward", "Kim", array("Jeju Univ", "Weird School"));
print $haruair;


// result :
// {
//   "full_name": "Edward, Kim",
//   "first_name": "Edward",
//   "last_name": "Kim",
//   "school": [
//       "Jeju Univ",
//       "Weird School"
//   ]
// }
?>

변환하는 과정에서 json 문자열을 추가하거나 데이터의 구조를 변환하는 등 다양한 형태로 활용할 수 있다.

더 읽을 거리

대부분 프로그램을 만족하고 사용하지만 불편하다고 느끼는 프로그램이 몇 있다.

Google Chrome

나는 크롬빠다. 하지만 요즘 메모리를 엄청나게 먹는 크롬을 깔 수 밖에 없다. 그 외에도 하나 있는데 Cmd + W를 한다는게 바로 옆에 있는 Q를 눌러 창을 전부 닫는 사태가 간혹 있다. 물론 크롬이 창을 잘 살려주기는 하지만 메모리 공룡이 된 이후 껐다 켜는데 생각보다 시간이 걸린다. Chrome에서 메뉴에 Chrome > Warn before quitting를 활성화 하면 창을 닫기 전에 경고를 준다는데 나는 동작하는 것을 본 적이 없는 것 같다. (발동 조건을 모르겠다. 될 때도 있고 안될 때도 있고.)

FileZilla

무료 FTP 프로그램으로 많은 사용자 정보를 저장해놓고 오랜 기간 사용하고 있다. 장점은 여러 플랫폼에서 사용이 가능하고 설정값을 플랫폼 상관없이 Export/Import 할 수 있다는 점이다. 여기에도 아쉽다고 느끼는 부분은 모든 단축키가 Windows 기준으로 되어 있어서 맥에서의 단축키를 바로 이용하지 못한다. 예를 들면 OSX의 Finder에서는 Enter로 파일명을 변경할 수 있지만 Filezilla에서 Enter를 치면 Windows 버전처럼 파일을 내려받게 된다.

MS Office for mac도 이도저도 아닌 애매한 단축키로 적응하는데 상당히 오랜 시간이 걸렸는데 FileZilla는 도무지 익숙해지지 않는다.

Adobe Photoshop CS6

개인 환경에는 Adobe CC를 사용하고 있지만 회사에서는 아직 CS6를 사용하고 있다. 강력한 사진/그래픽 도구로 사내에서 필요한 모든 디자인 작업을 Photoshop으로 해결하고 있다. 새로운 버전이 나와도 단축키는 거의 유지되는 편이라 크게 불편하지 않는데 CS6에서만 나타나는 이상한 문제점이 있다. 키보드를 영문이 아닌 한국어로 설정한 상태로 Photoshop을 실행하면 모든 단축키가 동작하지 않는다. 메뉴를 눌러 직접 종료를 한 후 영문으로 바꿔 다시 실행하면 되는데 이게 상당히 번거로운 작업이다.

처음에는 이 문제가 키보드 언어 선택이 이유라고 생각 해보지도 못했었다. 사용하다 보면 어쩌다 문제가 생기고 껐다, 켰다, 다른 작업을 하다가 다시 실행하면 문제가 해결되고 그랬었다.

Epson Scan

요즘 사진 촬영을 필름으로 주로 하는데 현상한 필름을 스캔할 때 Epson에서 기본적으로 제공하는 Epson Scan을 사용한다. 사실 이 프로그램은 2% 멀쩡하고 98% 부족한 프로그램이다. 일단 preset이 제대로 저장되지 않는다. 스캔 전에 preview를 하면 preset이 초기화되서 기본값으로 돌아가 매번 설정을 해줘야 한다. 그 덕분에 커브값을 설정한다거나 하는 과정은 스캔 후 후속 편집에서 하는데 그 덕분인지 관용도가 상당히 낮다. 그리고 일부 설정이 preset에 저장되지 않는다. Silverfast 구입을 고려하는 중이긴 하지만 비용이 상당히 커서 일단은 부족한대로 Epson Scan을 사용하고 있다. 이상하게 프린터, 스캐너 같은 하드웨어와 제공되는 프로그램은 한결 같이 불편한 사용자 경험을 제공하는 것 같다.


사소한 불편함이 전체적인 인상을 엉망으로 만들 때가 많다. 내가 만드는 서비스/웹사이트는 그런 일이 없도록 꼼꼼히 살펴봐야겠다.


이상한모임에서 진행하는, 다양한 주제로 함께 글을 쓰는 글쓰기 소모임입니다. 함께 하고 싶다면 http://weirdmeetup.herokuapp.com 에서 가입하시고 #weird-writing 채널로 오세요!

2014년에 다녀왔던 Global Windows Azure Bootcamp를 이번에도 다녀왔다. 작년에도 다녀 온 후기를 써야지 하고 쓰질 않았는데 이번엔 잊지 않고 적는다. Microsoft Windows Azure가 Microsoft Azure로 브랜딩이 변경되어 이번 행사명은 Global Azure Bootcamp가 되었다. 오늘 보고 들은 내용을 간단하게 정리한 포스트다. 들으면서 메모한 부분만 있어서 내용이 불친절 할 수 있다. MSDN이 있으면 좋고 없으면 Azure가 제공하는 Free Trial로도 충분히 둘러볼 수 있는 내용이다. GAB는 세계 각 지역에서 진행되고 있는데 아쉽게도 한국에서는 진행되고 있지 않는 것 같다.

함께 참석하기로 했던 모든 분들이 사정이 생겨 혼자만 참석하게 되었다. 9시 쯤 도착해서 티셔츠 받고 자리에 앉았다. 오늘은 ANZAC 데이라고 참전용사를 추모하는 호주 국경일이라서 키노트 시작 전에 관련 영상과 함께 추모 묵념을 했다.

시작 키노트에서는 간략하게 MS Azure에 대한 설명이 있었다. 현재 17개의 region을 제공하고 있고 region당 16개의 데이터 센터로 구성되어 있다고 한다. 데이터 센터 하나 당 풋볼 경기장, 보잉 747 2대 규모 정도 되어 물리적으로는 60만 대 가량의 서버가 있다고 한다. 현재 Azure에서 제공하는 서비스들을 소개했는데 지난번과 다르게 눈에 띄였던 부분은 Direct Access 였다. Azure의 VM을 사용하면 Azure를 거쳐 일종의 VPN과 같은 형태로 서버에 접속하게 되는데 이 경우 속도 등의 문제가 있었다고 한다. Direct Access는 지역 네트워크 제공자를 통해 지역 데이터 센터에 직접 접근하는 방식을 제공한다고 하는데 다른 곳에서 들어보지 못한 얘기 같아서 기억에 남았다.

그 외에도 다양한 서비스를 소개했는데 자세한 내용이 궁금하면 로드맵을 참고하라고 했고 Azure 관련 자격을 취득하고 싶으면 exam 70-532~4을 확인해보라고 한다. 늘 느끼는 것이지만 Azure 서비스는 전반적으로 브랜드 네이밍이 엉망으로 설명을 듣지 않고서는 정확하게 판단하기 어렵고 비슷한 서비스도 많아 복잡한 느낌이다.

이번 행사는 Developer Track과 IT Pro Track으로 구분해 각각 6개의 세션을 운영했다. 중간 중간 쉬는 시간까지 포함해 9시부터 5시까지 진행되었다. 장소는 Saxons였는데 Wifi가 자꾸 끊겨 인터넷이 연결 되어야만 하는 세션은 계속 새로고침 하고 기다리는 수 밖에 없어 아쉬웠다.

Azure App Services 1 – Websites and Mobile Services

깔끔한 새 azure portal과 함께 진행된 세션으로 Web App과 Azure Mobile App을 만들었다. 이전에는 훨씬 이상한 명칭이었는데 이제는 Web + Mobile 카테고리에서 쉽게 찾을 수 있다. 이 두 서비스는 PaaS로 ASP.Net MVC 프로젝트를 쉽게 올릴 수 있도록 돕는다. Azure Mobile App은 iOS, Android, Windows Mobile 등에서 백엔드로 사용할 수 있는 API를 쉽게 구성할 수 있도록 ASP.Net Web API로 만들어진 기본적인 코드를 제공한다.

Standard 이상의 요금 티어를 선택하면 Deployment Slots, Traffic Routing 등의 기능을 추가적으로 제공한다. 전자는 디플로이 할 수 있는 슬롯을 여러개 제공해 Staging과 Production 환경 구성을 돕는다. 후자는 이 PaaS 접근했을 때 어느 지역으로 연결해야 하는지 설정할 수 있다. 그 외에도 GitHub이나 Bitbucket도 지원하고 여러가지 세세한 설정이 많아져서 많이 편리해졌음을 알 수 있었다. 이 세션을 진행할 때 wifi 사정이 많이 안좋아 실습은 물론 진행까지 더뎌 아쉬웠다.

Getting started with Azure Operational Insights

Azure Operational Insights는 예전 Microsoft System Center Advisor 라는 이름으로 제공되던 서비스로 클라우드 기반의 분석 도구다. Windows 환경에서 사용할 수 있는 NewRelic라 볼 수 있는데 과거 System Center Operations Manager(SCOM)으로 확인할 수 있던 자료를 웹에서 바로 확인할 수 있도록 지원한다. MS Monitoring agent로 직접 접속하는 방법이 있고 SCOM 콘솔을 통해 접속하는 방법이 있다고 한다.

App services 2 Logic Apps and API Apps를 들을까 하다가 분석과 관련된 세션이라길래 가서 들었는데 생각과 많이 다르고 잘 알지 못하는 부분이었다.

Azure Storage Services

데이터를 저장할 수 있는데 사용할 수 있는 서비스인 Azure Storage Service는 Blobs, Tables, Queues, Files 4가지 방식으로 데이터를 저장할 수 있다. 이 4가지를 합쳐 account 당 500TB을 사용할 수 있고 azure 섭스크립션 하나 당 100 account가 허용된다.

  • Blobs: 사진, 음악, 비디오, 문서 등을 저장할 수 있고 여러 blob을 하나의 그룹으로 다룰 수 있도록 지원
  • Tables: noSQL과 같은 Key-value. 행 당 1MB이며 252개의 커스텀 어트리뷰트, 5개의 필수 어트리뷰트 지정 가능
  • Queues: 64KB 크기로 최소 1회 딜리버리를 보장
  • Files: 클라우드 기반 파일 공유 (Preview)

그리고 Document DB도 지원한다. Document-based NoSQL 이란 표현을 처음 들었는데 MongoDB 등이 Document DB라고 한다. 그 외에는 Azure Redis Cache도 지원한다. Azure에서 Storage Services 생성하면 예제 코드를 제공해서 쉽게 사용 가능하다.

누가 발표자에게 IaaS에 직접 설치하는 것에 비해 무엇이 장점인지를 물어봤다. 발표자가 ERP/CRM 관련 개발을 하는데 사용하고 있고 규모가 커져도 느려지지 않고 스케일링이 자유로워서 편리하다고 답변했다. 그 얘기를 듣고 좋은 서비스로 만들어서 괜찮은 API 레퍼를 제공해 사용자가 쉽게 사용할 수 있도록 만든 것은 박수칠 일이긴 하지만, 이 서비스가 아니고서 사용할 수 없는 형태라면, 서비스가 없어지거나 서비스 제약에 닿게 되면 생길 불편함도 염두해야 할 것 같다.

Building Apps Using Azure Active Directory

Azure Active Directory(Azure AD)를 적용해 만든 프로젝트를 시연했다. Azure AD는 기존에 있던 Active Directory를 온라인, 오프라인, MS에서 제공하는 오피스 365와 같은 클라우드 서비스, Azure AD 인증을 도입한 웹 어플리케이션 등에서 통합적으로 활용할 수 있도록 돕는다. 쉽게 설명하면 요즘 자주 보이는 페이스북 소셜 로그인 같은 기능을 Active Directory의 정보와 함께 사용할 수 있다는 뜻이다.(Single Sign On)

Visual Studio에서 ASP.Net Web Application 프로젝트를 생성하는 마법사에서 Change Authentication > Organizational accounts를 선택해 도메인 정보를 입력하면 바로 사용해볼 수 있다. 마법사를 사용하면 Azure AD 설정을 자동으로 생성해줘 기본적인 데이터를 입력하는 수고를 덜어준다. 이 구현은 wsfederation 프로토콜로 인증을 진행한다고 한다. Azure AD와 토큰을 주고 받고 토큰이 오면 사용자 정보를 사용할 수 있도록 바로 만들어준다.

OpenId로도 로그인이 가능하고 필요한 라이브러리는 다음과 같다.

Microsoft.Owin.Security
Microsoft.Owin.Security.Cookies
Microsoft.Owin.Security.OpenIdConnect

이 경우는 마법사가 제공하지 않는 방법이라 직접 개발해야 한다. ActiveDirectory 구현은 Microsoft.Owin.Security.ActiveDirectory를 활용할 수 있다.

다중 factor 인증도 지원한다. 시연에서는 전화, 문자 인증 두가지 방식을 보여줬는데 Azure AD와 연동도 깔끔한 SDK로 제공한다. 이 과정에서는 암호화를 위한 키를 필요로 했다. 이 기능으로 회원 인증과 같은 서비스를 활용할 수 있지 않을까 싶어 살펴봤는데 다중 factor 인증은 Azure AD 플랜 중 Premium에서만 제공하고 있었다. (사용자 당 $6 USD)

Azure AD의 예제는 Azure AD Samples 깃헙 페이지에서 확인할 수 있다.

Azure and Big Data

영화 Manhunt를 소개하면서 세션을 시작했다. 데이터를 수집하고 활용하는 일은 기존에도 있었지만 이전과 같은 접근으로 인사이트를 얻기에는 어렵기 때문에 더 방대하고 세세한 데이터를 수집해 가공/활용하는 방향으로 진화하고 있다고 한다.

최근의 추세는 Lambda Architecture로 데이터를 수집/가공을 한다고 한다. Batch layer, Speed layer, Serving layer 세가지로 구성된 이 아키텍쳐에 각각 필요한 라이브러리를 가지고 문장 분석을 하는 시연을 했다.

  • Batch layer: Storage (Hadoop, Azure Storage), Compute (Hadoop, Spark)
  • Speed layer: Storm, Spark Streaming, Azure stream analytics
  • Serving layer

Map Reduce로 Hortonworks 또는 Cloudera를 사용하는 방법과 HDInsight를 사용하는 방법이 있는데 시연은 HDInsight를 사용했다. HDInsight는 Hadoop as a service로 Azure Blob Storage에 데이터를 저장한다. HBase, Stork, Spark와 호환이 된다고 한다.

전반적인 라이브러리 추세를 다 설명한 덕분에 시간이 모자라 후반 시연은 진행하지 못했다. Hadoop은 많이 성숙했고 요즘은 Spark를 많이 사용한다고 한다.

IoT in Azure

다른 세션을 들으려고 했다가 자리 옮기기 귀찮아서 그냥 들었는데 재미있었다. Azure Event Hubs를 이용해 데이터를 수집하고 가공하는 것을 시연했는데 실제 사례를 기반으로 설명했다.

호주의 농장에서 트랙터와 같은 대형 장비를 사용하기 위해 오일 탱크를 가지고 있는데 이 탱크의 양을 수집해서 오일을 다시 주문해야 할 때 자동으로 주문하는 시스템을 개발했다고 한다. 호주의 네트워크 커버리지는 해안 지역 위주기 때문에 3G로 바로 전송을 할 수 없는 상황이고 현재로는 스마트폰이 다가가면 Bluetooth LE로 데이터를 수집하고 스마트폰이 3G 가능 지역으로 들어가면 그때 데이터를 클라우드로 올리는 방식으로 구현하고 있다고 한다.

Azure Event Hubs는 위와 같은 대량의 데이터 수집을 위한 이벤트 큐를 제공한다. 퍼블리셔가 허브로 이벤트를 보내면 이벤트를 Partition에 저장한 후 사용자에게 이벤트가 추가되었음을 호출한다. 파티션은 이벤트 규모에 따라 스케일링 할 수 있다. 사용자는 이 허브를 통해 변동값만 받는 형태로 구성되어 있고 허브에 쌓인 이벤트는 GC를 통해 자동으로 정리된다. 즉 허브가 앞에서 본 Batch layer의 Storage를 담당하고 사용자가 Compute를 하게 된다.

사용자의 구현은 event Process host model과 event receiver model로 구현할 수 있고 파티션 당 프로세서가 어떻게 배당되는가에 따라 구분된다. 후자는 각각의 파티션마다 receiver가 있는 형태고 전자는 여러 파티션이 하나의 프로세스에 배당되는 방식이다.

Event Hubs는 Azure에서 Data Analytics 항목에서 찾을 수 있다.

시연 중에 데이터 시각화에는 Power BI를 활용했는데 깔끔했다.

뒷얘기

작년에 했던 장소에 비해 좁고 Wifi 환경이 별로 좋지 않았다. 시연에 불편할 정도였는데 최소 발표자는 안정적으로 네트워크를 사용할 수 있는 상황을 제공해야 했다. 그래도 발표 사이사이 버퍼로 둔 시간이 커서 시간이 밀리거나 하진 않았다.

모닝티, 점심, 오후 간식 세번을 했는데 각각의 식단에서 글루틴 무첨가, 채식과 할랄 식단을 제공했다. 이제 이런 배려는 당연하게 느껴진다.

이번에도 행운권 추첨을 했는데 잘 모르는 프로그램 라이센스를 받았다. Azure Bootcamp는 항상 뽑히는 것 같은데 다음엔 DDD에서 서피스 프로 같은걸 받았으면 좋겠다. (만족할 줄 모르는 남자.)

Microsoft Virtual Academy를 대대적으로 홍보하고 있었다. 확실히 MS에서 열심히 밀고 있는 느낌이다. C# 관련 몇 강의를 수강해봤는데 내용도 괜찮고 재미있게 따라갈 수 있었다.


작년에 살펴본 Azure에 비해 훨씬 깔끔해졌고 세세한 서비스가 많아졌다. 여전히 서비스명이 복잡한 느낌이지만 새 포털에서 사용하면 쉽게 찾아서 사용할 수 있게 많이 개선되었다. AWS에 비해 free tier를 크게 홍보하지 않는게 각각 서비스마다 요금 책정 방식이 다르기 때문인 것 같다. 어떤 서비스는 기본이 무료고 어떤 서비스는 시작부터 비용을 청구해서 그런 것 같은데 Azure Websites나 Azure AD Free 등은 무료로 제공하고 있으니 살펴보는 것도 좋겠다.

다음에도 기회가 있으면 참석하고 싶다. 그때는 Azure도 사용하고 그래서 이것저것 물어볼 부분이 많았으면 좋겠다.

PHP Package Checklist의 번역 글이다. 패키지 개발을 하지 않고 있더라도 PHP 개발을 하고 있다면 충분히 염두해볼 만한 내용이 포함되어 있고 참고할 이야기가 많다.

패키지명을 현명하게 선택하기

  • 다른 프로젝트에서 사용되고 있지 않은 이름을 선택한다.
  • 패키지명과 PHP 네임스페이스가 일치하도록 관리한다.
  • 성이나 개인 닉네임을 PHP 네임스페이스로 사용하지 않는다.

소스를 공개적으로 호스팅하기

  • 공개 프로젝트는 GitHub를 무료로 사용할 수 있다.
  • GitHub은 이슈를 관리하고 기능 요청이나 풀 리퀘스트에 도움이 된다.
  • 대안으로 Bitbucket도 사용할 수 있다.

Autoloader 친화적으로 개발하기

  • PSR-4와 호환이 되는 네임스페이스를 사용한다.
  • 코드는 src 폴더 내에 넣는다.

Composer를 통해 배포하기

  • PHP를 위한 의존성 관리 도구인 Composer에서 라이브러리를 사용 가능하게 한다.
  • Composer의 주 리포지터리인 Packagist에 등록한다.

프레임워크에 대해 독립적으로 개발하기

  • 프로젝트를 하나의 프레임워크에 제한을 두지 않는다.
  • 서비스 프로바이더를 제공해 특정 프레임워크에서 사용할 수 있도록 지원한다.

코딩 스타일을 따르기

  • PSR-2 코딩 스타일 가이드를 철저히 지킬 것을 강력하게 권장한다.
  • 빠르게 자동으로 코드를 수정해주는 PHP Coding Standards Fixer를 사용한다.
  • 코딩 표준에 대해 자동으로 확인해주는 PHP Code Sniffer를 사용한다.

유닛 테스트를 작성하기

  • 주요 코드를 커버하는 것에 초점을 둔다.
  • PHPUnit은 사실상 표준인 PHP 유닛 테스트 프레임워크다.
  • 대안으로 phpspec, Behat, atoum, Codeception이 있다.

DocBlock을 사용하기

  • 인라인 문서화를 위해 DocBlock을 제공한다.
  • DocBlock은 PhpStorm과 같은 IDE의 코드 완성을 향상하는데 도움이 된다.
  • phpDocumentor를 활용해 API 문서로 자동 변환이 가능하다.

유의적 버전을 사용하기

  • 버전 번호를 관리하는데 유의적 버전을 사용한다.
  • 주버전.부버전.수버전(MAJOR.MINOR.PATCH) 시스템을 사용한다.
  • 개발 버전의 업그레이드는 변경으로 인해 깨지는 것을 걱정하지 않도록 안전하게 제공해야 한다.
  • 릴리즈마다 tag로 버전을 적는 것을 잊지 말자.

변경 로그를 유지하기

  • 매 릴리즈를 할 때마다 변경 로그를 깔끔하게 공개한다.
  • Keep a CHANGELOG 양식을 사용하는 것을 고려한다.

지속적인 통합(continuous integration)을 사용하기

  • 자동으로 코딩 표준과 테스트를 구동하는 서비스를 사용한다.
  • 다양한 버전의 PHP에서 테스트를 구동하는 좋은 방법이다.
  • 풀 리퀘스트가 제출될 때 자동으로 구동할 수 있다.
  • Travis-CI, scrutinizer, circleci를 사용할 수 있다.

상세한 문서를 작성하기

  • 좋은 문서화는 성공적인 패키지에 필수적인 요소다.
  • 적어도 설명을 포함한 README를 작성해 리포지터리에 포함한다.
  • 문서를 GitHub Pages로 제공하는 것을 고려한다.
  • 대안으로 Read the Docs를 활용할 수 있다.

라이센스를 포함하기

  • 라이센스를 포함하는 것은 현재까지 한 일을 보호할 수 있는 작은 방법이다.
  • choosealicense.com 사이트를 참고한다. 대부분의 PHP 프로젝트는 MIT 라이센스를 사용한다.
  • 적어도 LICENSE 파일은 라이브러리에 포함해야 한다.
  • Dockblock에도 라이센스를 포함할 것을 고려해본다.

기여를 환영하기

  • 당신의 프로젝트를 누군가 돕길 원한다면 당연히 그걸 물어봐야 한다.
  • CONTRIBUTING 파일을 사용해 프로젝트 기여를 환영하자.
  • 이 파일에 테스트와 같은 프로젝트 요구 사항을 설명하는 내용을 작성한다.

위 리스트의 모든 내용을 한번에 적용할 수 없다면 필요한 부분부터 점차적으로 적용해 나가도록 하자. Autoloader는 익숙해지면 include 지옥에서 벗어날 수 있는 강력한 기능이다. 위에서 소개된, 코딩 스타일을 자동으로 교정해주는 도구들은 문법 고민을 덜어주고 비지니스 로직에 집중할 수 있도록 돕는다. 현대적인 PHP 개발을 생각하고 있다면 위 모든 항목 하나하나 살펴보는 것이 도움이 된다.

더 읽을 거리

Micro-framework의 전성기라고 할 만큼 다양한 환경과 언어로 프레임워크가 쏟아지고 있다. PHP에도 micro-framework가 많이 나와 있는데1 최근 Laravel에서 Lumen을 발표했다. 발표 자료에서는 symfony2 기반인 silex보다 1.9배 빠르다고 하는데 문법적으로는 Silm과 상당히 유사한 느낌도 든다. 기존에 나왔던 프레임워크와 엄청나게 큰 구조 차이를 가지고 있는 것은 아니지만 Laravel과의 호환을 염두한 부분도 많다는 느낌을 받았다. 또한 구조적으로도 silex나 여타 기존에 나온 micro-framework 보다 훨씬 깔끔하고 미려하다는 느낌을 받았다.

lumen logo

이 포스트는 lumen 문서에서 쉽게 볼 수 있는 부분만 다뤘고 더 깊은 내용을 보고 싶다면 Lumen 공식 문서를 보는게 도움이 된다. Lumen는 PHP >= 5.4를 요구하며 Mcrypt, OpenSSL, mbstring, tokenizer 확장을 필요로 한다.

Lumen 설치

lumen을 설치하기 위해서는 composer가 설치되어 있어야 한다.

lumen을 사용해 프로젝트를 시작하는 방법은 lumen installer를 사용하는 방법과 composer의 create-project로 생성하는 방법이 있다. 결과물은 동일한데 installer 속도가 더 빠르다.

composer.json, phpunit.xml 등 단순한 스캐폴딩을 함께 제공한다.

Lumen Installer

다음 명령어로 Lumen Installer를 설치한다. 이 installer는 커맨드라인 환경에서 lumen 프로젝트를 시작할 수 있도록 기능을 제공한다.

composer global require "laravel/lumen-installer=~1.0"

설치가 완료되면 lumen new 명령어로 프로젝트를 생성할 수 있다. 여기서 helloworld라는 이름으로 프로젝트를 생성했다.

lumen new helloworld

다음과 같이 프로젝트가 생성된 것을 확인할 수 있다.

lumen init

composer

위 인스톨러를 사용할 수 없다면 composer create-project 생성할 수 있다.

composer create-project laravel/lumen --prefer-dist

설정하기

lumen은 laravel과 다르게 모든 설정을 .env에 저장한다. 쉽게 활용할 수 있도록 .env.example 템플릿이 제공된다. 데이터베이스, 캐시, 큐, 세션과 관련한 설정값을 지정할 수 있다. 설정에서 가장 먼저 할 일로 해당 템플릿을 열어 APP_KEY에 32자 무작위 문자열을 입력한 후 .env로 저장한다.

URL 설정

Apache 환경을 사용하고 있다면 public/.htaccess를 활용할 수 있고 Nginx에서는 다음과 같이 설정할 수 있다.

location / {
    try_files $uri $uri/ /index.php?$query_string;
}

HTTP 라우팅

라우팅 기초

route는 app/Http/routes.php에 작성한다. 여타 micro-Framework과 크게 다르지 않은 문법이다.

$app->get('/', function() {
  return 'Hello World';  
});
$app->post('foo/bar', function() {
  // do something
});
$app->patch('foo/bar', function() {
  // do something
});
$app->put('foo/bar', function() {
  // do something
});
$app->delete('foo/bar', function() {
  // do something
});

이렇게 생성한 route에서 URL을 다른 곳에서 사용하고 싶다면 url 헬퍼를 쓴다.

$list_link = url('foo');

router에서 파라미터는 다음과 같이 사용한다.

$app->get('user/{id}', function($id) {
  return 'User '.$id;
});

$app->get('user/{name:[A-Za-z]+}', function($name) {
  // do something
  // 이 문법은 라라벨과 호환되지 않는다.
});

컨트롤러와도 쉽게 연결할 수 있다.

$app->get('user/{id}', 'UserController@showProfile');

위 route와 같이 복잡한 URL 구조를 가지고 있다면 route에 as로 별칭을 지정해 쉽게 활용할 수 있다.

$app->get('user/profile', ['as' => 'profile', function() {
  // show user profile
}]);

$app->get('user/dashboard', [
  'as' => 'dashboard',
  'uses' => 'UserController@showDashboard'
]);

이제 위 route는 profile이라는 이름으로 활용할 수 있다. 파라미터가 있는 경우는 두번째 파라미터에 array로 값을 넣으면 된다.

$url = route('profile');
$redirect = redirect()->route('profile');

$profile_url = route('profile', ['id' => 1]);

라우팅 그룹 묶기

route를 그룹으로 묶어 미들웨어나 네임스페이스를 지정할 수 있다. 여기에서 $app->group()를 활용한다.

Closure를 기반으로 한 미들웨어는 다음과 같이 사용할 수 있다.

$app->group(['middleware' => 'foolbar'], function($app) {
  $app->get('/', function() {
    // do something
  });
  $app->get('user/profile', function() {
    // do something
  });
});

namespace로 특정 네임스페이스에 있는 컨트롤러를 불러 활용할 수 있다.

$app->group(['namespace' => 'Admin'], function() {
  // "App\Http\Controllers\Admin" 네임스페이스에 있는 컨트롤러
});

HTTP 예외 발생하기

abort 헬퍼를 이용한다. 응답을 같이 보내줄 수도 있다.

abort(404);
abort(403, 'Unauthorised action.');

이 헬퍼는 상태 코드와 함께 Symfony\Component\HttpFoundation\Exception\HttpException 예외를 던진다.

뷰는 resources/views에 php 파일로 저장한다.

<!-- View stored in resources/views/greeting.php -->

<!doctype html>
<html>
    <head>
        <title>Welcome!</title>
    </head>
    <body>
        <h1>Hello, <?php echo $name; ?></h1>
    </body>
</html>

다음과 같이 사용할 수 있다.

$app->get('/', function() {
  return view('greeting', ['name' => 'James']);
});

$app->get('/admin', function() {
  /* ... */

  // resources/views/admin/dashboard.php
  return view('admin.dashboard', $data);
});

데이터 바인딩을 배열로 넘길 수 있지만 with 메소드나 매직 메소드를 활용할 수도 있다.

$view = view('greeting')
          ->with('name', 'Edward')
          ->with('age', 20)
          ->withLocation('Jeju'); // 매직 메소드

컨트롤러

규모가 커지면 routes.php 파일 하나로만 로직을 다루는 것보다 컨트롤러를 활용해서 구조화하는 것이 낫다. 컨트롤러로 HTTP 요청을 조작하는 로직을 쉽게 묶을 수 있다. 컨트롤러는 app/Http/Conterollers에 저장한다.

컨트롤러는 App\Http\Conterollers\Controller 기초 클래스를 필수적으로 상속해야 한다. 기본적인 컨트롤러는 다음과 같다.

<?php namespace App\Http\Controllers;

use App\User;
use App\Http\Controllers\Controller;

class UserController extends Controller {

    /**
     * Show the profile for the given user
     *
     * @param  int  $id
     * @return Response
     */
    public function showProfile($id)
    {
        return view('user.profile', ['user' => User::findOrFail($id)]);
    }

}

UserController를 라우터에서 다음과 같이 연결할 수 있다.

$app->get('user/{id}', 'App\Http\Controllers\UserController@showProfile');

의존성 주입

Lumen과 Laravel의 서비스 컨테이너는 타입 힌트를 통해 의존성을 해결해준다. 생성자 주입, 메소드 주입 둘 다 사용 가능하다. 다음 코드는 생성자의 타입 힌트 UserRepository, store 메소드의 타입힌트 Request로 의존성이 주입되는 예제다.

<?php namespace App\Http\Controllers;

use App\Http\Controllers\Controller;
use App\Repositories\UserRepository;
use Illuminate\Http\Request;

class UserController extends Controller {

    /**
     * The user repository instance.
     */
    protected $users;

    /**
     * Create a new controller instance.
     *
     * @param  UserRepository  $users
     * @return void
     */
    public function __construct(UserRepository $users)
    {
        $this->users = $users;
    }

    /**
     * Store a new user.
     *
     * @param  Request  $request
     * @return Response
     */
    public function store(Request $request)
    {
        $name = $request->input('name');
    }

}

미들웨어

HTTP 미들웨어는 HTTP 요청과 응답을 제어할 수 있도록 돕는 구조로 micro-Framework에서는 흔히 사용되고 있다. (Python의 uWSGI, .Net의 OWIN) 미들웨어는 app/Http/Middleware에 위치한다.

미들웨어를 활용하기 위해 구성해야 하는 메소드는 handle이다.

<?php namespace App\Http\Middleware;

class OldMiddleware {

    /**
     * Run the request filter.
     *
     * @param  \Illuminate\Http\Request  $request
     * @param  \Closure  $next
     * @return mixed
     */
    public function handle($request, Closure $next)
    {
        if ($request->input('age') < 200) {
            return redirect('home');
        }

        return $next($request);
    }

}

미들웨어로 HTTP 요청과 응답을 모두 제어할 수 있는데 $next($request)를 기준으로 그 앞에서는 요청을 제어하고 뒤에서는 응답을 제어할 수 있다.

다음은 요청을 제어하는 미들웨어로 이 내용을 실행한 후 어플리케이션에 접근하게 된다.

<?php namespace App\Http\Middleware;

class BeforeMiddleware implements Middleware {

    public function handle($request, Closure $next)
    {
        // Perform action

        return $next($request);
    }
}

다음은 응답을 제어하는 미들웨어 예시로 어플리케이션에서 처리가 끝난 후 클라이언트에게 전달되는 응답을 제어할 수 있다.

<?php namespace App\Http\Middleware;

class AfterMiddleware implements Middleware {

    public function handle($request, Closure $next)
    {
        $response = $next($request);

        // Perform action

        return $response;
    }
}

Service Provider와 Service Container

Lumen은 Service ProviderService Container로 기초를 구성하고 있다. Service Provider는 어플리케이션을 시작하기 전에 준비해야 할 작업을 처리할 수 있다. bootstrap/app.php를 열어보면 $app->register() 메소드를 확인할 수 있는데 이 메소드를 통해 추가적인 service provider를 등록할 수 있다.

Service Container는 클래스 간의 의존성을 해결하기 위한 도구로 생성자 또는 “setter” 메소드를 통해 의존성을 주입해준다. $app->bind(), $app->singleton()을 통해 resolver를 등록할 수 있다. 컨테이너에서 의존성을 주입 받기 위해서는 $foobar = $app->make('FooBar'); 방식으로 make 메소드를 사용하는 방법이 있고, 앞서 살펴본 방식인 생성자나 개별 메소드에서 타입 힌팅을 이용해 의존성을 주입할 수 있다. 자세한 내용은 각 문서를 참고하자.


Micro-framework지만 그 말이 무색할 만큼 현대적인 PHP 개발에서 필요한 필수적인 요소는 모두 포함된 강력함을 보여주고 있다. 이 포스트에서 자세하게 다루진 않았지만 많은 서비스도 제공하고 있으며 기존에 laravel을 사용하고 있다면 더 간편하게 사용할 수 있을 것이라 본다. 지금까지 나온 micro-framework 중에서는 가장 마음에 드는 구성이라 차기 프로젝트에서 사용하는 것을 생각해보고 있다. 앞으로 기대가 많이 되는 프레임워크다.

  • PHP 기반의 Micro Frameworks 정리 
  • PHP 5.3에서 새로운 기능으로 네임스페이스가 추가되었다. (= 이미 오래된 기능이다.) 많은 현대 언어는 이미 이 기능을 추가한지 오래지만 PHP는 조금 늦게 추가되었다. 최근에 개발되는 대다수의 PHP 라이브러리는 네임스페이스로 패키징해 composer, League 등을 통해 제공되고 있어 현대 PHP를 사용하려고 한다면 필수적으로 알아야 하는 기능이다.

    PHP에서는 같은 이름을 가진 두 클래스를 동시에 사용할 수 없다. 클래스는 항상 유일해야만 한다. 이 제한으로 인해 서드파티 라이브러리에서 User라는 클래스명을 사용하고 있을 때는 User라는 클래스명을 사용할 수 없었다. 이렇게 간단한 클래스명을 사용하지 못하는건 불편하다.

    PHP 네임스페이스는 위와 같은 클래스명 중복 문제를 해결한다. 그 뿐 아니라 코드를 패키징하거나 벤더명을 지정해 소유권을 표시하는데도 사용할 수 있다.

    전역 네임스페이스

    여기 간단한 클래스가 있다.

    <?php
    class Edward
    {
    
    }
    

    별로 특별한 부분이 없다. 다음과 같이 사용할 수 있다.

    <?php
    
    $edward = new Edward();
    

    이 상황에서 이 클래스는 전역 네임스페이스를 가졌다고 볼 수 있다. 즉 이 클래스는 네임스페이스 없이 존재한다. 그냥 일반 클래스와 같다.

    단순한 네임스페이스

    이제 네임스페이스 밑으로 클래스를 만들어보자.

    <?php
    namespace Haruair;
    
    class Edward
    {
    
    }
    

    위에서 만든 클래스와 유사하지만 작은 차이가 있다. namespace 라는 지시문이 추가되었다. namespace Haruair;는 여기서 작성한 모든 PHP 코드가 Haruair 네임스페이스와 관련이 되어 있음을 뜻하고 이 파일에서 생성한 클래스가 모두 Haruair 네임스페이스에 포함되어 있음을 뜻한다. 네임스페이스를 통해 클래스를 생성하면 다음과 같이 사용할 수 있다.

    <?php
    $edward = new Haurair\Edward();
    

    위 코드와 같이 네임스페이스와 함께 클래스를 선언할 수 있다. 네임스페이스와 클래스 사이에는 백슬래시()로 구분이 된다. 위와 같은 방법으로 네임스페이스로 클래스를 다룰 수 있다.

    이와 같은 방법으로 여러 단계의 위계를 활용하고 있는 경우를 많이 볼 수 있다.

    This\Namespace\And\Class\Combination\Is\Silly\But\Works
    

    의존성 원칙

    PHP는 현재의 namespace에 따라 상대적으로 동작한다.

    <?php
    namespace Haruair;
    
    $edward = new Edward();
    

    Haruair 네임스페이스 내에서 개체를 생성했다. 동일한 네임스페이스에 속해 있는 상황이기 때문에 Haurair\EdwardEdward로 호출해 사용할 수 있다.

    이런 상황에서 반대로 생각해볼 수 있는 부분은 네임스페이스 내부에서 상위 또는 최상위에 있는 네임스페이스나 클래스는 어떻게 접근할 지에 대해서다.

    PHP는 클래스명 앞에 백스래시()를 넣어 전역 클래스 또는 글로벌 네임스페이스를 사용하고 있음을 명시적으로 선언할 수 있다.

    <?php
    $edward = new \Edward();
    

    만약 다른 네임스페이스에 속한 클래스인 Drink\CokeHaruair 네임스페이스 내에서 사용한다면 앞서 예제와 같이 작성할 수 있다.

    <?php
    namespace Haruair;
    
    $coke = new \Drink\Coke();
    

    매번 전체 위계를 입력하는 것이 번거롭다면 use를 활용할 수 있다.

    <?php
    namespace Haruair;
    
    use Drink\Coke;
    
    $coke = new Coke();
    

    use를 활용하면 다른 네임스페이스에 있는 하나의 클래스를 현재 네임스페이스 내에서 사용할 수 있게 해준다. 동일한 클래스명을 불러오게 되는 경우가 온다면 다음과 같이 활용할 수 있다.

    <?php
    namespace Haruair;
    
    use Drink\Pepsi as BlueCoke;
    
    $pepsi = new BlueCoke();
    

    위와 같이 as 키워드를 쓰면 Drink\Pepsi 클래스에 별칭 BlueCoke를 지정해 사용할 수 있다. 같은 이름의 클래스 여럿을 동시에 사용한다 해도 문제 없다.

    <?php
    namespace Facebook;
    
    use Twitter\User as TwitterUser;
    
    class User {}
    
    $twitter_user = new TwitterUser();
    $facebook_user = new User();
    

    Twitter 네임스페이스에 있는 UserTwitterUser 별칭으로 불러오면서 충돌을 회피했다. 이와 같이 충돌을 피하고 의도와 필요에 따라 기능을 모아서 사용할 수 있다.

    use는 필요한 만큼 넣어서 사용할 수 있다.

    <?php
    namespace Haruair;
    
    use Twitter\Follower;
    use Facebook\WallPost;
    use Cyworld\WallPost as CyPost;
    

    구조

    네임스페이스는 단순히 충돌을 피하기 위해서만 사용하는 것이 아니라 조직이나 소유권을 표기하기 위해 사용하기도 한다.

    오픈소스 라이브러리를 만든다고 가정하자. 내가 만든 코드를 다른 사람이 사용한다면 분명 좋을 것이다. 다만 내 코드를 사용하는 사람들에게 불편함을 주지 않았으면 좋겠다. 클래스명이 충돌하게 되면 엄청나게 불편할 것이 확실하다. 그래서 다음과 같이 네임스페이스를 구분하기로 했다.

    Haruair\Blog\Content\Post
    Haruair\Blog\Content\Page
    Haruair\Blog\Tag
    

    여기서 내 아이디를 사용해서 이 코드가 누가 만들었는지 표시하는 것과 동시에 내 라이브러리 안에 만들어 코드를 사용하고자 하는 사람의 코드와 충돌하지 않도록 돕는다. 내 기초 네임스페이스에 여러개의 서브 네임스페이스를 만들어 내부 구조를 잡았다.

    Composer를 사용하면 PSR-0, PSR-4를 통해 정해진 규칙에 따라 네임스페이스를 통해 클래스 정의를 자동으로 불러오는 등의 작업을 할 수 있다. 위 두 문서에서 이 유용한 방식을 확인해보는 것을 강력하게 추천한다.

    제한

    PHP가 제공하는 네임스페이스에는 한계가 있다. 다른 언어들에서의 구현과는 거의 유사하지만 약간 다른 점이 존재한다. Java의 경우, wildcard(*)를 이용해 해당 네임스페이스에 속해 있는 모든 클래스를 한번에 불러올 수 있다. 또한 Java에서의 import는 앞서 살펴 본 use와 같은 역할을 해서 패키지나 네임스페이스 내에 있는 클래스를 쉽게 이용할 수 있게 돕는다. 다음은 Java의 예시다.

    import haruair.blog.*;
    

    위와 같은 코드로 haruair.blog의 모든 패키지를 로드할 수 있다.

    PHP는 이와 같은 방법으로 불러올 수 없다. 대신 상위의 네임스페이스를 use로 불러와 유사하게 사용할 수 있다.

    <?php
    namespace weirdmeetup;
    
    use Haruair\Blog as Cms;
    
    $post = new Cms\Content\Post;
    $page = new Cms\Content\Page;
    $tag = new Cms\Tag;
    

    한 네임스페이스에서 많은 클래스를 동시에 사용할 때 위 방법이 도움이 된다.

    더 읽을 거리

    생산성 도구를 안 쓰는 사람은 있어도 하나만 써보는 사람은 없다는 얘기가 있다. 생산성 도구를 사용하는 사람이라면 거기서 거기인 앱이 계속 나오는 기분이 들겠지만 하나씩 사용해보면 각자가 독특한 개성을 보여주고 있어 자신에게 가장 맞는 도구를 찾기 위해 여럿을 사용하게 된다는 얘기다. 그만큼 생산성 도구는 개인화 경향이 강한 분야다.

    나는 무료로 제공되는 마크다운 에디터를 쓰다가 Ulysses 추천을 받고 Ulysses를 구입해 한동안 사용했다. 강력한 기능을 많이 지원하는 도구지만 md 파일로 바로 저장되는 식이 아니라 자체 라이브러리에서 관리되는 형태에서 불편함을 느꼈고 그 이후로는 만능 에디터 Sublime Text를 사용하고 있었다. Sublime Text를 쓰면서 사실 큰 불편은 없지만 내가 좋아하는 Typewriter 모드가 없었다. 1

    Typed

    Typed는 지난번 9to5mac에서 Screenflow와 Things2 등과 번들을 구입할 때 같이 포함되어 있었다. 이 번들이 Things2를 단일로 구입하는 것보다 저렴하길래 구입했었다. 매년 GTD를 새해 선물처럼 구입하는데 한 달을 못간다. 아무튼 기대하고 구입한 Things2 보다는 Typed를 더 열심히 사용하고 있다. (No strings attached.)

    사운드 트랙 내장

    에디터 리뷰에서 이 얘기를 맨 처음 하는게 웃기긴 한데 에디터에 8가지 사운드 트랙이 내장되어 있다. 들어보면 차분해질 만 한 음악을 끊임 없이 반복적으로 틀어준다. 같은 노래를 오랜 기간 반복적으로 듣는 편이라서 알게 모르게 흥얼거리고 그러다보면 다른 생각에 빠지게 되는데 이 사운드 트랙은 흥얼거릴 거리도 없는 조용한 음악을 반복해서 들려준다. 그러다 보니 심지어 이 에디터를 코딩할 때도 켜놓고 그 음악을 틀어놓기도 한다.

    Typewriter 모드

    Typed도 Typewriter 모드를 지원해서 타자기처럼 글이 작성되는 위치를 중앙에 고정해놓고 사용할 수 있다. 에디터에서 글을 길게 작성하면 입력하는 커서 위치가 화면 가장 밑에 있게 되는데 이걸 방지해준다. 커서 밑에 공간이 없으면 답답한 기분도 들고 계속 글을 작성하는게 어색한 느낌이 드는데 Typewriter 모드로 그 공간을 만들어줄 수 있다.

    사실 이 기능은 대부분의 마크다운 에디터에서 지원하고 있다. 게다가 Ulysses에서는 이 기능에 더 많은 옵션을 지원하고 있는데도 Typed에서의 구현이 더 마음에 든다. 많은 옵션을 제공하는 것 보다 딱 맞는 옵션으로 하나만 지원하는 것은 사용자의 입장에서 더 배려받는 기분을 들게 한다.

    Little big things

    문단/문장 강조 기능도 편리하다. 기본적으로 문장 강조 기능을 사용하고 있는데 현재 커서가 위치한 문장 외에는 옅은 색상으로 변경해준다. 글을 다 작성하고서 문장 또는 문단 단위로 리뷰할 때 편리하다.

    글자, 단어 수 표시도 깔끔하다. 우측 상단에 글자 수, 단어 수 표시가 있는데 눈에 거슬리지 않고 단순하다.


    Typed Main

    생산성 관련 서비스/앱을 개발하는 것이 힘들다는 얘기를 듣게 된 이후 생산성과 관련된 앱을 하나씩 사용하게 될 때마다 작은 부분까지도 유심히 살펴보려고 노력하고 있다. 안쓰는 사람은 안쓰고 쓰는 사람은 여럿을 사용한다는 생산성 앱에서 자신에게 딱 맞는 앱을 찾는다. 반대로 개발하는 입장에서 생각한다면 좁은 마켓에 있는 더 작은 타겟을 대상으로 딱 맞는 앱을 만들어야 하는데 사용자층이 좁을 수록 freemium이 아닌 이상 가격을 산정하기가 쉽지 않을 것 같다.

    그 사용자층을 넓히기 위해서 세세하게 제어 가능한 옵션을 제공할 수도 있겠다. 단순하게 생각하면 세세한 옵션이 사용자에게 편의를 준다고 생각할 수 있지만 반대로 너무 많은 옵션에 혼란이 올 수 있다. 그건 마치 칫솔 회사의 상황과도 비슷하다. 사용자가 칫솔모를 자신의 구강 구조에 맞게 다듬어서 사용하는 킷을 파는게 아니라 사용자에게 미리 최적화 된, 잘 닦일 수 있는 칫솔을 디자인해서 팔아야 하는 것이다. 사용자는 앱을 칫솔처럼 구입해서 사용한다. 본인이 직접 설정을 잘못해놓고는 앱이 불편하다고 여기고 떠날 수도 있다. 사용자에게 “모든 설정을 제공하고 스스로 trial & error을 통해 자신에게 최적의 환경을 만든다” 가정은 유니콘 찾기일 수도 있다. 최고의 설정은 개발사가 정해서 제공해야 한다. 물론 개발사가 타겟한 사용자 층에 대해 전문가가 되는 것은 말처럼 쉽지 않겠지만 말이다.

    앞으로도 나에게 잘 맞는 생산성 앱을 만날 수 있었으면 하는 기대가 있다. 비싸더라도 구입할 수 있도록 열심히 잔고를 늘려놔야겠다.

  • 왠지 플러그인으로 있을 법도 한데 설치하고 코딩, 글쓰기 스위칭 할 때마다 끄고 켜야 한다면 없는게 낫다. 
  • 몰입에 대해 어렵지 않다고 생각하고 지내왔고 실제로도 쉽게 몰입하는 경향이 있어서 처음엔 좋은 주제라고 느꼈는데 막상 작성하려고 하니 최근에는 오랜 시간을 몰입해본 기억이 없었다. 쉽게 몰입했던 그 감각이 다시 살아났으면 하는 마음에서 요 며칠은 퇴근하고 집중적으로 번역 글을 올리는데 시간을 사용했다.

    몰입을 방해하는 환경

    나는 몰입에 큰 어려움이 없는 편이었다. 하지만 최근 들어 흐름을 끊는 방해 요소가 많아졌다. 한두 번 정도야 다음 몰입에 영향을 주지 않지만 이 상황이 반복되면 그 상황에 스트레스를 받는다. 반복적으로 집중이 틀어지면 또 집중이 흐트러질 것이라는 사실을 몸이 무의식적으로 예상하고 있어서 그런지 다시 몰입 상태로 진입하기 힘들다. 그러면 일도 지루하게 느껴지고 재미도 없어진다.

    이렇게 스트레스를 받는 상황에서는 주변 환경에 대해 배로 민감해진다. 누군가의 대화, 문 여닫는 소음, 전화벨 소리, 발소리, 기침 소리, 심지어는 작은 노티피케이션 하나에도 방해로 느껴지고 스트레스를 더 받는 악순환에 빠진다. 작은 자극에도 불편함을 느끼고 있다면 무언가 잘못되고 있음을 자각해야 한다. 공간을 개선하거나, 공간을 탈출하거나. 참으면 본인만 손해다. 그래서 요즘은 몰입하는 것 자체보다 자연스럽게 몰입할 수 있는 환경을 만드는데 고민을 많이 하고 있다.

    몰입의 대상과 조건

    당연한 점인지 몰라도 나는 내가 좋아하는 일을 할 때 몰입을 잘한다. 재밌는 일을 좋아한다. 처음 하거나 아직 익숙지 않아서 서툰 일을 재밌어한다. 숙련도가 늘어나는 반복 작업도 재밌다. 없는 것을 만드는 작업을 재밌다. 이미 있는 것을 개선하는 작업도 재밌다. 내가 단순해서 그런지 웬만하면 다 재미있어한다.

    재미있으려면 호기심이 생겨야 하고 호기심이 있으려면 여유가 있어야 한다. 마음에 여유가 없으면 눈앞에서 치워버리는 것이 목표가 된다. 급박한 기일을 가진 데드라인을 정하고 달리는 것도 좋지만 이런 긴장감이 반복 되다보면 결국 고무줄은 늘어나고 사람은 지치고 여유의 종말을 맞이하게 된다. 쉴 때 확실하게 쉬지 않으면 여유가 생기지 않는다. 반복되는 데드라인에도 지치지 않는 사람은 애초에 내가 생각하는 범위 이상으로 여유의 크기가 남다르게 큰 사람이거나 쉴 때 엄청나게 잘 쉬는 사람이다. 이런 사람들은 내적 요인, 외적 요인 가릴 것 없이 쉽게 몰입해서 과정을 즐긴다.

    내게 몰입에 도움이 되는 공간은 주변은 어둡지만 내 손 닿는 곳은 밝은, 그리고 밀폐된 느낌이 있는 공간이다. 즉 도서관 열람실 같은 분위기에서 쉽게 몰입한다. 오랫동안 집중해서 무엇인가 한 기억은 모두 그와 비슷한 공간이라 그런지 그와 같은 환경에서 몰입이 더 잘된다. 어쩌면 학습된 결과인지도 모르겠다.

    노래를 듣거나 TV를 보면서는 절대 안된다. 가사 없는 곡은 좀 도움이 되는 것 같다. 조용한 공간에서 집중이 잘되는 편이다. 화이트 노이즈 같은 에어컨 돌아가는 소음이나 컴퓨터 돌아가는 소음 있는 공간도 잠만 잘 오고 집중에 방해된다.

    소음 얘기하다 생각났는데 세상엔 두 종류의 사람이 있다고 한다. 방을 정리하는 사람과 방을 정리하지 않는 사람인데 전자는 뭘 해도 바로 정리하는 타입이고 후자는 정리를 아예 하지 않거나 어쩌다 한번 정리하는 타입이다. 나는 후자인 편인데 어쩌면 물건이나 옷이 많아서 소음을 줄여주는 차음재 역할을 하는 건 아닐까 생각이 든다. 1 아무튼 나에겐 번잡스러운 공간도 집중에 도움이 된다.

    몰입 해야 하나

    몰입의 순간에 행복감을 느낀다. 몰입의 대상이 무엇이냐에 따라 상관 없이 몰입이라는 경험 자체에 중독성이 있다. 성취로 인한 기쁨도 있지만 내가 얼마나 집중해서 결과를 만들었는지도 만족의 척도가 된다. 결과는 썩 좋지 않더라도 몰입하고 나면 “괜찮아, 나는 꽤 즐겁게 한 걸” 하고 기분 좋게 넘어갈 수 있다. 힘들 수 있는 과정을 즐겁게 수행하는 비결이 된다.

    반대로 몰입에서 오는 만족감 때문에 몰입에 대한 강박관념에 사로잡히기 쉽다. 일을 잘 끝내고 나서도 몰입하지 못했다는 이유로 허전한 기분에 빠지거나 일을 망쳤다는 생각에 사로잡힐 수 있다. 개운하지 않은 기분. 이것도 은근 스트레스다. 몰입은 필수가 아니다. 과정에서 부가적으로 얻을 수 있는 즐거움인데 몰입 자체에 집착하면 강박증세만 심화된다.

    몰입은 자신의 환경을 돌아봐야 한다는 신호다. 몰입이 안되는 상황을 자신 탓이라 생각하면 그건 해결될 수 없는 방식이다. 사람은 다 다르고 다 다른 방식의 몰입 트리거를 가지고 있을 텐데 다 같은 방법으로 몰입할 수 있을까. 누구는 너저분한 책상에서 집중이 잘 되고 누구는 깨끗한 책상에서 집중이 잘 된다. 몰입하지 못한다고 답답해하기보다 어떤 환경에서 내 여유를 챙기고 충전할 수 있는지 고민하고, 어떤 환경에서 몰입할 수 있는지 찾는 것이 더 중요하다.


    이상한모임에서 진행하는, 다양한 주제로 함께 글을 쓰는 글쓰기 소모임입니다. 함께 하고 싶다면 http://weirdmeetup.herokuapp.com 에서 가입하시고 #weird-writing 채널로 오세요!

  • 방정리 잘 안하는 사람의 흔한 변명. 
  • 운영하는 사이트의 외부 유입을 확인하기 위해 Google Analytics를 기본적으로 설치하는 편이다. Google Analytics는 설치만 해도 유입 트래픽을 보기 좋게 정렬해서 보여주는 편이지만 좀 더 세부적으로 데이터를 구분하기 위해서는 몇가지 추가적인 작업이 필요하다. 웹사이트로의 외부 유입을 추적하는 경우에는 맞춤 캠페인(Custom campaign)을 활용해 필요한 데이터만 분리해 확인할 수 있다. 이 기능이 도움 될 만한 시나리오를 생각해보면 다음과 같다.

    • SNS에서 유입되는 트래픽 중 내가 직접 작성한 포스트로 유입된 트래픽만 확인
    • 광고를 내는데 이 광고로 들어오는 사람이 얼마나 되는지 확인
    • 뉴스레터에서 어느 링크를 클릭해 웹사이트로 유입되는지 확인

    맞춤 캠페인 URL

    맞춤 캠페인에 필요한 것은 맞춤 캠페인 URL이다. 이 URL을 통해 접속하면 URL에 포함된 매개변수를 활용해 분류할 수 있다. 맞춤 캠페인에서는 다음 5개의 매개변수를 활용해서 분류할 수 있다.

    • utm_source: 사이트로 트래픽을 보내고 있는 광고주, 사이트, 출판물 등을 식별 (예: Google, 도시검색, 뉴스레터4, 빌보드)
    • utm_medium: 광고 또는 마케팅 매체 (예: CPC, 배너, 이메일 뉴스레터)
    • utm_campaign: 제품의 개별 캠페인 이름, 슬로건, 프로모션 코드 등
    • utm_term: 유료 검색 키워드. 유료 키워드 캠페인에 대해 직접 태그를 추가할 경우 utm_term을 사용하여 키워드를 지정
    • utm_content: 같은 광고 내에서 유사 콘텐츠 또는 링크를 구분. 예를 들어 하나의 이메일 메시지에 두 가지 클릭 유도문안 링크가 있는 경우 utm_content를 사용하여 각각 다른 값을 설정하면 어떤 버전이 더 효과적인지 확인할 수 있음.

    추후 편리하게 리포트를 확인하려면 일관성을 가지고 입력해야 한다. 예를 들면 대소문자를 섞어쓰지 않도록, 소스(source)와 매체(medium)를 혼동해 뒤바꿔 쓰지 않도록, 시즌별로 반복된다면 연월 등 규칙성이 있는 포맷으로 작성하는게 도움이 된다.

    캠페인 URL 만들기

    이 맞춤 캠페인 URL은 URL 작성도구로 쉽게 만들 수 있다. Google Analytics에서 결과를 확인하기 위해 이 도구로 맞춤형 캠페인 URL 예제를 하나 만든다.

    URL 작성도구

    Google Analytics에서 확인하기

    작성한 주소를 통해 접속을 하면 Google Analytics 에서 해당 정보와 함께 수집된다. Google Analytics는 접속하더라도 정보 수집이 바로 반영되지 않아 디버깅 하기가 힘들었는데 이제는 실시간 접속을 확인할 수 있어서 예전보다 간편하다. 확인을 위해 위 URL 작성도구에서 연습으로 만든 맞춤형 캠페인 URL로 접속해서 창을 열어둔 채 다음 순서대로 Google Analytics에서 확인해보자.

    1. Google Analytics에 접속
    2. 자신의 사이트를 선택한 후
    3. 왼쪽 탭에서 **실시간(Real-Time) > 트래픽 소스(Traffic Source)**를 클릭

    트래픽이 URL 작성도구에서 입력한 데이터와 함께 정상적으로 들어오는 것을 확인할 수 있다.

    실시간 유입 확인

    이렇게 유입된 트래픽은 **획득(Acquisition) > 캠페인(Campaigns) > 모든 캠페인(All Campaigns)**에서 종합된다.

    모든 캠페인

    직접 맞춤설정을 통해 맞춤 보고서를 만들어서 데이터를 확인하는 것도 가능하다. 더 자세한 내용을 확인하고 싶으면 맞춤 캠페인 – 웹로그 분석 도움말을 확인하자.

    msdn 블로그에 게시된 New Features in C# 6포스트를 요약했다. C# 6는 VS 2015 프리뷰와 함께 제공된 버전으로 여러가지 문법 특징이 추가되었다. 이 포스트는 요약이라 내용이 좀 부실할 수 있는데 상세한 내용은 위 포스트를 참고하자.

    표현식

    nameof

    새로운 형태의 문자열로 프로그램 요소의 이름을 간혹 알아내야 할 상황을 마주하는데 그때 사용할 수 있는 표현식이다. 점 표기법(dot notation)을 사용한 경우 가장 마지막 식별자를 반환한다.

    if (x == null) throw new ArgumentNullException(nameof(x));
    WriteLine(nameof(person.Address.ZipCode)); // prints "ZipCode"
    

    person이 스코프 내에서 타입이 아닌 변수라면 두번째 코드는 허용되지 않는다.

    문자열 인터폴레이션

    번거로웠던 String.Format()을 간편하게 작성할 수 있다.

    var s = String.Format("{0} is {1} year{{s}} old", p.Name, p.Age);
    s = "\{p.Name} is \{p.Age} year{s} old";
    s = "\{p.Name,20} is \{p.Age:D3} year{s} old"; // 형식 지정
    s = "\{p.Name,20} is \{p.Age:D3} year{(p.Age == 1 ? "" : "s")} old"; // 표현식도 쓸 수 있음
    

    Note. 프리뷰 이후 문법이 변경되었다. s = $"{p.Name,20} is {p.Age:D3} year{{s}} old";

    Null 조건부 연산자

    체이닝 등 호출이 연속적으로 이뤄지는 상황에서 null 확인을 더 쉽게 만드는 연산자다.

    int? length = customers?.Length; // customers가 null이면 null 반환
    Customer first = customers?[0]; // customers가 null이면 null 반환
    
    // null 병합 연산자인 ??와 함께 사용. customers가 null 일 때, 값은 0
    int length = customers?Length ?? 0;
    
    // 뒤에 따라오는 멤버 접근, 엘리먼트 접근 등은 customers가 null이 아닐 때만 호출
    int? first = customers?[0].Orders.Count();
    
    // 위와 동일한 표현
    int? first = (customers != null) ? customers[0].Orders.Count() : null;
    
    int? first = customers?[0].Orders?.Count(); // 연속으로 사용 가능
    

    다만 ?을 사용한 직후에는 문법적으로 모호함이 있어서 바로 호출을 할 수 없다. 그래서 바로 대리자를 호출할 경우 다음과 같이 작성해야 한다.

    if(predicate?(e) ?? false) { ... } // 에러 발생
    if(predicate?.Inkobe(e) ?? false) { ... }
    

    이벤트를 작동할 때 다음과 같이 작성할 수 있다.

    PropertyChanged?.Invoke(this, args);
    

    이 문법은 스레드 안정성을 보장하는데 좌측을 평가한 후 값을 임시로 저장하기 때문이다.

    인덱스 이니셜라이저

    개체 이니셜라이저를 확장에 다음과 같이 인덱스를 넣을 수 있게 되었다. 표현식 하나로 JSON 개체를 만들 때 유용하다.

    var numbers = new Dictionary<int, string> {
      [7] = "seven",
      [9] = "nine",
      [13] = "thirteen",
    };
    

    확장 Add 메소드

    확장 Add 메소드를 컬렉션 이니셜라이저에서 사용할 수 있다. 예전엔 인스턴스 메소드만 Add를 호출할 수 있었다.

    오버로드 향상

    오버로드 확인(resolution)이 향상되었다. 상세 내역은 소개되지 않았고 nullable 값 타입을 받을지 말지, 메소드 그룹을 대리자로 받는 등의 향상이 있을 것이라고 한다.

    표현문

    예외 필터

    CLR 호환으로 VB와 F#에만 있던 기능이 이제 추가되었다.

    try { ... }
    catch (MyException e) if (myfilter(e))
    {
      ...
    } // if 문이 참일 때만 `catch` 블럭이 실행된다.
    

    필터는 일반적이고 수용할 수 있는 형태로 “오용”을 할 수 있다. 파생작업을 위해 다음과 같이 사용할 수 있다.

    private static bool Log(Exception e) { /* log it */; return false; }
    ...
    try { ... }
    catch (Exception e) if (Log(e)) {}
    

    catch/finally 블럭에서의 Await

    Resource res = null;
    try
    {
        res = await Resource.OpenAsync( ... ); // 원래 되던 부분
    }
    catch(ResourceException e)
    {
      await Resource.LogAsync(res, e); // 이제 가능한 부분
    }
    finally
    {
      if (res != null) await res.ColseAsync(); // 여기서도 가능
    }
    

    맴버 선언

    자동 프로퍼티 이니셜라이저

    필드에 이니셜라이져 하는 것과 비슷하다. 이 방법으로 이니셜라이징 하면 setter를 거치지 않고 내부 형식 바로 저장이 된다.

    public class Customer
    {
      public string First { get; set; } = "Jane";
      public string Last { get; set; } = "Doe";
    }
    

    Getter only 자동 프로퍼티

    setter 없이 자동 프로퍼티를 사용하는게 허용된다. 이 방식은 readonly로 암묵적인 선언이 된다.

    public class Customer
    {
      public string First { get; } = "Jane";
      public string Last { get; } = "Doe";
    }
    

    위와 같이 초기화 하지 않는 경우에는 다음과 같이 타입 생성자에서 선언하면 값이 내부 형식에 바로 저장된다.

    public class Customer
    {
      public string Name { get; };
      public Customer(string first, string last)
      {
        Name = first + " " + last;
      }
    }
    

    이 문법은 표현 타입을 더 간소하게 만든다. 하지만 변형 가능한 타입과 불변 타입의 차이가 없어진다. 변형 가능한 클래스도 상관이 없다면 자동 프로퍼티를 기본으로 사용하는 것도 좋다. 여기다 getter only 자동 프로퍼티는 변형 가능한 타입과 불변 타입을 더 비슷하게 만든다.

    표현-본문 함수 멤버

    표현-본문 함수 멤버는 메소드와 프로퍼티, 다른 종류의 함수 멤버들의 본문을 블럭이 아닌 람다처럼 표현식을 바로 쓸 수 있도록 지원한다. 실제로 람다처럼 람다 화살표로 작성한다. 블럭 본문에 단일 반환값을 가진 형태와 동일하다.

    public Point Move(int dx, int dy) => new Point(x + dx, y + dy);
    public static Complex operator +(Complex a, Complex b) => a.Add(b);
    public static implicit operator string(Person p) => $"{p.First} {p.Last}";
    

    void를 반환하는 메소드, Task를 반환하는 비동기 메소드에서도 동일하게 사용할 수 있지만, 이 경우에는 람다에서와 같이 꼭 문 표현식(statement expression)이 사용해야 한다.

    public void Print() => Console.WriteLine(First + " " + Last);
    

    프로퍼티와 인덱서도 다음과 같이 쓸 수 있다. get 키워드가 없는 대신 표현식 본문 문법에 따라 암묵적으로 사용된다.

    public string Name => First + " " + Last;
    public Customer this[long id] => store.LookupCustomer(id);
    

    파라미터 없는 구조체 생성자

    파라미터 없는 구조체 생성자가 허용되었다. 다음 구조체에서 new Person()은 선언된 생성자에 따라 기본값을 제공한다. default(Person)로 기본값을 사용하거나 new Person[...] 형태로 배열을 사용하면 해당 생성자는 실행되지 않는다. 명시적으로 구조체 타입과 함께 new 키워드를 사용했을 때만 해당된다.

    struct Person
    {
      public string Name { get; }
      public int Age { get; }
      public person(string name, int age) { Name = name; Age = age; }
      public Person() : this("Jane Doe", 37){ }
    }
    

    임포트

    using static

    using 으로 정적 멤버 타입을 스코프에서 직접 사용할 수 있다. 프리뷰에서는 정적 클래스의 멤버만 불러올 수 있다.

    using System.Console;
    using System.Math;
    
    class Program
    {
      static void Main()
      {
        WriteLine(Sqrt(3*3 + 4*4));
      }
    }
    

    Note. 다음과 같이 디자인이 변경되었다.

    1. using에서 using static으로 변경
    2. 구조체나 enum과 같은 멤버의 비정적 타입을 임포트 할 수 있음
    3. VB 모듈의 멤버나 F#의 최상위 함수를 임포트 할 수 있음
    4. 최상위 스코프에서 확장 메소드는 임포트 할 수 없음. 확장을 해온 원 클래스는 불러오지 않고 확장 메소드만 불러오면 안되기 때문.

    위 모든 기능들은 VS 2015 프리뷰에서 확인할 수 있다.


    이 글을 뒤늦게 확인하고 C#6의 문법적인 변경점을 살펴봤다. (아직 기본적인 문법도 잘 모르긴 하지만.) C# 개발은 얼마 해보지 못했는데 문자열 인터폴레이션만 봐도 편리한 기능들이 많이 나오는구나 느낄 수 있었다. 이번 달 아니면 다음 달 내로 베타가 시작될 것 같은데 기대된다.

    색상을 바꿔요

    눈에 편한 색상을 골라보세요 :)

    Darkreader 플러그인으로 선택한 색상이 제대로 표시되지 않을 수 있습니다.