목차

CCleaner 자동 업데이트 스크립트 제작

동기

'CCleaner (Crap Cleaner)'를 아시나요? 윈도우를 사용하다보면 불필요한 레지스트리, 임시파일, 휴지통 파일 등등… 소위 말하는 '찌꺼기'들이 쌓이기 시작합니다. PC 좀 만지시는 분들은 상당히 꺼리는 것들이죠. 이런 것을 관리해주는 최적화 툴이 여럿 있는데, CCleaner도 이런 PC 최적화 툴 중 하나입니다. 간단하고 꽤 안정적이어서 저는 오랫동안 즐겨 사용하고 있습니다.

CCleaner는 개인이 집에서 무료로 이용할 수 있는 '홈(Home)', 좀 더 기능이 강화된 '프로페셔널(Professional)', 그리고 '비즈니스(Business)' 등 몇 가지 버전이 있습니다. 무료로 이용할 수 있는 홈 버전도 사실 상당히 쓸만합니다. 다만 자동 업데이트가 지원되지 않아 매번 홈페이지에서 직접 파일을 다운로드 받아 설치를 해 줘야 한다는 점이 불편합니다.

그런데 쓰다 보면 이 무료 버전의 자동 업데이트 미지원이 은근히 귀찮습니다. 대략 2-3주, 길면 한 두달에 한 번 생각날 때마다 한 번 정도 사용하는데 거의 이 주기가 지나면 적어도 1번쯤은 프로그램의 업데이트가 되어 있죠. 그래서 사실상 CCleaner를 사용할 때마다 먼저 업데이트부터 완료한 후에야 최적화 작업을 하곤 했습니다. 이 푸닥거리를 수회 반복하면 “자동 업데이트가 지원되는 유료를 질러 볼까?” 생각을 해보게 됩니다. 네, 지르셔도 됩니다. 좋다고 생각하시면 하나 지르세요 ;-)

하지만 여전히 무료로 사용하면서, 이 귀찮음을 온전히 감수하시는 분들이 계실 터입니다. 그런 분들을 위해 살짝 반칙 같지만, 그래서 약간 제작사인 Piriform에 미안한 맘이 들지만, 할 수 있는 적합한 선에서 이 귀찮음을 덜 수 있는 방법이 없을까 생각해 보았습니다. 그리고 제 파이썬 잉여력을 좀 쏟아보기로 했습니다.

사실 얼마전부터 CCleaner의 유료 버전인 CCleaner Professional을 사용중입니다. 이 버전은 자동 업데이트가 지원되어 편하게 사용할 수 있지만, 사실 무료 버전의 CCleaner와 별다른 차이가 없습니다. 자동 업데이트가 지원되지 않는 점만 감안하면 무료 버전도 충분히 쓸만합니다. 사실 그렇게 큰 차이도 없는데 돈을 지불한 것이 조금 아깝기까지도 합니다. 이번에도 역시 파이썬 2.7 스크립트를 사용합니다. CCleaner는 윈도우 기반의 소프트웨어이므로 이 스크립트는 윈도우에서만 동작한다고 가정합니다.

작업 방법

크게 아래와 같은 순서로 작업하면 될 것 같습니다.

  1. 로컬 PC의 CCleaner.exe의 file version을 쿼리한다.
  2. Piriform에서 CCleaner의 파일 버전을 쿼리한다.
  3. 로컬 버전의 파일 버전이 낮다면 새로 다운로드 받아 자동설치한다.

세부적으로 각 단계를 구상해 보겠습니다.

로컬 PC 버전 쿼리

CCleaner의 버전은 CCleaner의 파일 속성을 조사해보면 쉽게 알 수 있습니다. 탐색기에서 CCleaner.exe 파일을 선택해 우측 마우스를 클릭해서 '속성'을 클릭하면 나오는 아래 그림과 같은 창에 나오는 정보입니다.

문제는 이 정보를 우리가 쉽게 프로그램으로 가져올 수 있도록 해야합니다. 이렇게 마우스 클릭 및 메뉴 선택 등의 작업이 들어가면 자동화하기 어렵습니다. 이러한 명령을 CLI 상에서 쉽게 할 수 있는 방법을 찾아야 합니다. 아주 다행히도 이러한 방법은 구글에서 쉽게 검색을 통해 알아냈습니다. 이렇게 하면 됩니다.

wmic DATAFILE WHERE NAME="C:\\Program Files\\CCleaner\\CCleaner.exe" GET version > version.txt
 
[version.txt 파일]
Version
4.0.0.4064

WMIC는 MSDN에서 자세하게 설명하고 있습니다. 여러 관리 문서 및 예제등은 검색하면 많이 찾아내실 수 있습니다. 여기서 저는 이 CCleaner의 파일 버전만 알아내고 싶을 뿐이므로 자세한 내용은 지나가도록 하겠습니다. 이렇게 얻어낸 version.txt 파일을 파이썬으로 처리하는 것은 식은 죽 먹기입니다.

온라인 최신 버전 쿼리

최신 버전 정보는 공식 홈페이지의 웹페이지를 크롤링해서 얻어낼 수 있습니다. 이 부분이 사실 민감하다면 민감한 부분입니다. 물론 홈페이지에서 버전 정보를 얻어내는 것이 문제가 되는 내용은 아닙니다만, 홈페이지의 레이아웃은 언제든 변경될 수 있으니까요. 레이아웃이 변경되면 그 전까지는 잘 동작하던 스크립트가 갑자기 먹통이 될 수도 있습니다. 그때는 어쩔 수 없습니다. 직접 레이아웃에 맞추어 수정을 해야 합니다. 무료의 한계입니다. :-( Piriform이 너무 자주 레이아웃을 바꾸지는 말아 주었으면 합니다.

최신 버전은 이 곳의 'Release notes'를 통해 공지됩니다. 이 부분의 HTML 코드는 다음과 같이 되어 있습니다. (2013년 4월 기준)

<h2 class="icon_edit">Release notes</h2>
<div class="indent">
<ul class="versionHistory">
    <li>
          <strong>
          v4.00.4064
          </strong> (26 Mar 2013)
          .....

매우 쉽습니다. ul class=“versionHistory”는 이 페이지에서 단 한 번 나옵니다. 그럼 여기서 strong 태그 사이의 버전과 릴리즈된 날짜를 가져오도록 하면 되겠군요.

파일 다운로드 및 설치

이 부분도 Piriform이 디자인을 바꾸면 그에 따라 가야 하는 부분입니다. 다운로드는 이 곳에서 받을 수 있습니다.

위 그림처럼 'restart the download' 글자 부근의 a 태그를 따라 가면 ccleaner 설치 파일을 쉽게 받을 수 있을 것입니다.

이 다음은 새롭게 받은 CCleaner를 설치해야합니다. 그런데 여기서 GUI 창을 통해 클릭을 할 수는 없지요. 다운로드가 끝남과 동시에 자동으로 설치가 되어야 합니다. CCleaner 무료 버전도 다행히 명령줄을 통한 자동 설치가 가능합니다. 다음과 같은 명령을 입력하면 됩니다.

ccsetup.exe /S /L=1042 /D=<pathname>

'/L=1042' 스위치는 한글로 설치하라는 옵션입니다. 타 언어는 이 페이지에서 확인하세요. '/D='는 유저가 원하는 곳에 설치할 때 필요한 옵션입니다. '/S'는 기본값으로 알아서 설치하기 위해 사용됩니다. 크롬과 같은 광고 프로그램들이 이 스위치를 켠다고 해서 같이 설치되지는 않는 것으로 파악하였습니다.

구현

설정 파일

일단 간단한 텍스트 파일로 된 설정 파일을 먼저 만들도록 하겠습니다. 설정 파일의 내용에는 아래 사항이 기록될 것입니다.

ccleaner_update.cfg
ccleaner_path: C:\\Program Files\\CCleaner\\CCleaner.exe
release_url: http://www.piriform.com/ccleaner/download
relaese_re: <ul class="versionHistory">.+<strong>(.+)</strong> \((.+?)\)
download_url: http://www.piriform.com/ccleaner/download/standard
download_re: <a href="(.+)">restart the download</a>
install_arg: /S /L=1042

본 구현

CCleanerAutoUpdate.py
# -*- coding: UTF-8 -*-
import sys
import os
import urllib2
import re
 
def GetHTML(URL):
	obj  = urllib2.urlopen(URL)
	html = obj.read()
	obj.close()
	return html
 
# Version 확인을 위한 클래스
class Version:
 
	major = None
	minor = None
	build = None
 
	# 로컬 버전 문자열 읽음.
	def FromLocalVersionString(self, localVersionString):
		vers = localVersionString.split('.')
		self.major = int(vers[0])
		self.minor = int('%s%s' % (vers[1], vers[2]))
		self.build = int(vers[3])
 
	# 최신 버전 (웹페이지에서 확인한) 문자열 읽음
	def FromCurrentVersionString(self, currentVersionString):
		vers = currentVersionString.split('.')
		self.major = int(vers[0])
		self.minor = int(vers[1])
		self.build = int(vers[2])
 
	def __str__(self):
		return '%d/%d/%d' % (self.major, self.minor, self.build)
 
	# 로컬 버전 <= 최신버전이므로 !=, == 만 정의.
	def __eq__(self, other):
		return isinstance(other, self.__class__) and self.__dict__ == other.__dict__
 
	def __ne__(self, other):
		return not self.__eq__(other)		
 
 
def CheckUpdate(config):
	# check CCleaner.exe exists!
	if os.path.exists(config['ccleaner_path']) == False:
		print >> sys.stderr, '%s does not exist!' % config['ccleaner_path']
		return 1
 
	# local ccleaner version	
	print "Checking installed CCleaner's version..."
	command = 'wmic DATAFILE WHERE NAME="%s" GET version > version.txt' % config['ccleaner_path']
	os.system(command)
 
	# get version
	with open('version.txt', 'r') as f:
		text   = unicode(f.read(), 'utf-16')
		iv_txt = text.split(u'\n')[1].strip()		
 
	# current release version
	print 'Checking current CCleaner version at Piriform...'
	release_html = GetHTML(config['release_url'])
	release_exp  = re.compile(config['release_re'] , re.DOTALL|re.MULTILINE)
	release_srch = release_exp.search(release_html).groups()
 
	cv_txt  = release_srch[0].strip()[1:] # exclude heading 'v'
	reldate = release_srch[1].strip()
 
	# To canonical versino expression.
	installed_ver = Version()
	current_ver  = Version()
 
	installed_ver.FromLocalVersionString(iv_txt)
	current_ver.FromCurrentVersionString(cv_txt)
 
	print 'Installed CCleaner version:', installed_ver
	print 'Current CCleaner version:', current_ver, reldate
 
	# compare two
	if installed_ver == current_ver:
		print 'You\'re using current version. Nothing to do.'
		return 0	
	else:
		print 'There\'s a new update available!'
 
	# download current release
	download_html = GetHTML(config['download_url'])
	download_exp  = re.compile(config['download_re'])
	download_srch = download_exp.search(download_html)
 
	url           = download_srch.groups()[0]
	filename      = url.split('/')[-1]		
 
	print 'Begin downloading...'
	obj = urllib2.urlopen(url)
	with open(filename, 'wb') as f:
		f.write(obj.read())
		obj.close()
 
	print 'Downloading complete!'
 
	cmd = filename + ' ' + config['install_arg']
 
	print 'Installing...'
	os.system(cmd)
	print 'Complete!'	
	return 0
 
def ParseConfig(configfile):
	config = {}
	with open(configfile, 'r') as f:
		for l in f:
			colon = l.find(':')
			prop  = l[0:colon].strip()
			val   = l[colon+1:].strip()
 
			config[prop] = val
 
	return  config
 
def main(argv):	
 
	# parse config file
	configfile = './CCleanerAutoUpdate.cfg'
	if len(argv) == 2:
		configfile = argv[1]
	config = ParseConfig(configfile)
 
	# begin the job
	return CheckUpdate(config)
 
if __name__ == '__main__':
	sys.exit(main(sys.argv))

끝마치며

참으로 잉여롭습니다! :-) 혹시나 CCleaner 설치라는 혹을 떼려다가 스크립트 관리라는 혹을 붙인 게 아닌가 싶습니다. 하지만 호기심을 충족했다는 점에서는 대만족입니다. :-) 계속 사용하다보면 ccsetup.exe 파일이 쌓이게 될 수도 있는데, 이것은 설치 후 직접 삭제하시든지, 스크립트를 수정하시면 됩니다. 관리자 권한까지는 완전 자동화하기 어려웠습니다. 이건 사용자가 한 번 확인해 주어야 할 부분이네요. 스크립트는 윈도우 8에서 동작을 확인했습니다.

참고자료