[Claude Code Desktop 자동 설치 환경]
- setup/CLAUDE.md: 트리거 키워드 + 설치 패키지 설명
- setup/.claude/skills/guardia-install/SKILL.md: 6단계 설치 오케스트레이터
Phase 0: 의도 파악 → Phase 1: OS 감지 → Phase 2: 사전 확인
Phase 3: 설치 실행 → Phase 4: 라이선스 발급 → Phase 5: 검증 → Phase 6: 완료보고
[통합 자동 설치 스크립트]
- setup/install_auto.sh: Linux 통합 (OS 자동 감지 ubuntu/centos/rhel)
- --license trial30|trial7|<key> 파라미터
- 설치 완료 후 GUARDiA 자동 실행 + 브라우저 자동 열기
- --test 검증 모드
- setup/install_auto.ps1: Windows 통합 (ASCII 전용, PS 5.1 호환)
- 설치 후 NSSM 서비스 자동 시작 + 브라우저 자동 열기
- -Test 파라미터로 검증 전용 실행
[라이선스 엔진 개선]
- core/license.py: generate_trial_key(days=None) 파라미터 추가
- TRIAL_DURATION_DAYS = TRIAL_DURATION_DAYS 환경변수로 조정 가능
- routers/license.py: TrialRequest.days 필드 + 30일 체험판 지원
POST /api/license/trial {"days": 30} 로 30일 발급
사용자 경험:
1. setup/ 폴더를 새 PC에 복사
2. Claude Code Desktop 열고 해당 폴더 open
3. "GUARDiA 시스템 1달 사용자로 설치해 줘" 입력
4. 자동으로 OS 감지 → 설치 → 30일 라이선스 → 브라우저 열림
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
162 lines
4.4 KiB
JavaScript
162 lines
4.4 KiB
JavaScript
'use strict'
|
|
|
|
let { existsSync, readFileSync } = require('fs')
|
|
let { dirname, join } = require('path')
|
|
let { SourceMapConsumer, SourceMapGenerator } = require('source-map-js')
|
|
|
|
function fromBase64(str) {
|
|
if (Buffer) {
|
|
return Buffer.from(str, 'base64').toString()
|
|
} else {
|
|
/* c8 ignore next 2 */
|
|
return window.atob(str)
|
|
}
|
|
}
|
|
|
|
class PreviousMap {
|
|
constructor(css, opts) {
|
|
if (opts.map === false) return
|
|
if (opts.unsafeMap) this.unsafeMap = true
|
|
this.loadAnnotation(css)
|
|
this.inline = this.startWith(this.annotation, 'data:')
|
|
|
|
let prev = opts.map ? opts.map.prev : undefined
|
|
let text = this.loadMap(opts.from, prev)
|
|
if (!this.mapFile && opts.from) {
|
|
this.mapFile = opts.from
|
|
}
|
|
if (this.mapFile) this.root = dirname(this.mapFile)
|
|
if (text) this.text = text
|
|
}
|
|
|
|
consumer() {
|
|
if (!this.consumerCache) {
|
|
this.consumerCache = new SourceMapConsumer(this.json || this.text)
|
|
}
|
|
return this.consumerCache
|
|
}
|
|
|
|
decodeInline(text) {
|
|
let baseCharsetUri = /^data:application\/json;charset=utf-?8;base64,/
|
|
let baseUri = /^data:application\/json;base64,/
|
|
let charsetUri = /^data:application\/json;charset=utf-?8,/
|
|
let uri = /^data:application\/json,/
|
|
|
|
let uriMatch = text.match(charsetUri) || text.match(uri)
|
|
if (uriMatch) {
|
|
return decodeURIComponent(text.substr(uriMatch[0].length))
|
|
}
|
|
|
|
let baseUriMatch = text.match(baseCharsetUri) || text.match(baseUri)
|
|
if (baseUriMatch) {
|
|
return fromBase64(text.substr(baseUriMatch[0].length))
|
|
}
|
|
|
|
let encoding = text.slice('data:application/json;'.length)
|
|
encoding = encoding.slice(0, encoding.indexOf(','))
|
|
throw new Error('Unsupported source map encoding ' + encoding)
|
|
}
|
|
|
|
getAnnotationURL(sourceMapString) {
|
|
return sourceMapString.replace(/^\/\*\s*# sourceMappingURL=/, '').trim()
|
|
}
|
|
|
|
isMap(map) {
|
|
if (typeof map !== 'object') return false
|
|
return (
|
|
typeof map.mappings === 'string' ||
|
|
typeof map._mappings === 'string' ||
|
|
Array.isArray(map.sections)
|
|
)
|
|
}
|
|
|
|
loadAnnotation(css) {
|
|
let comments = css.match(/\/\*\s*# sourceMappingURL=/g)
|
|
if (!comments) return
|
|
|
|
// sourceMappingURLs from comments, strings, etc.
|
|
let start = css.lastIndexOf(comments.pop())
|
|
let end = css.indexOf('*/', start)
|
|
|
|
if (start > -1 && end > -1) {
|
|
// Locate the last sourceMappingURL to avoid pickin
|
|
this.annotation = this.getAnnotationURL(css.substring(start, end))
|
|
}
|
|
}
|
|
|
|
loadFile(path, cssFile, trusted) {
|
|
/* c8 ignore next 5 */
|
|
if (!trusted && !this.unsafeMap) {
|
|
if (!/\.map$/i.test(path)) {
|
|
return undefined
|
|
}
|
|
}
|
|
this.root = dirname(path)
|
|
if (existsSync(path)) {
|
|
this.mapFile = path
|
|
return readFileSync(path, 'utf-8').toString().trim()
|
|
}
|
|
}
|
|
|
|
loadMap(file, prev) {
|
|
if (prev === false) return false
|
|
|
|
if (prev) {
|
|
if (typeof prev === 'string') {
|
|
return prev
|
|
} else if (typeof prev === 'function') {
|
|
let prevPath = prev(file)
|
|
if (prevPath) {
|
|
let map = this.loadFile(prevPath, file, true)
|
|
if (!map) {
|
|
throw new Error(
|
|
'Unable to load previous source map: ' + prevPath.toString()
|
|
)
|
|
}
|
|
return map
|
|
}
|
|
} else if (prev instanceof SourceMapConsumer) {
|
|
return SourceMapGenerator.fromSourceMap(prev).toString()
|
|
} else if (prev instanceof SourceMapGenerator) {
|
|
return prev.toString()
|
|
} else if (this.isMap(prev)) {
|
|
return JSON.stringify(prev)
|
|
} else {
|
|
throw new Error(
|
|
'Unsupported previous source map format: ' + prev.toString()
|
|
)
|
|
}
|
|
} else if (this.inline) {
|
|
return this.decodeInline(this.annotation)
|
|
} else if (this.annotation) {
|
|
let map = this.annotation
|
|
if (file) map = join(dirname(file), map)
|
|
let unknown = this.loadFile(map, file, false)
|
|
if (unknown) {
|
|
try {
|
|
/* c8 ignore next 4 */
|
|
this.json = JSON.parse(unknown.replace(/^\)]}'[^\n]*\n/, ''))
|
|
} catch {
|
|
return undefined
|
|
}
|
|
}
|
|
return unknown
|
|
}
|
|
}
|
|
|
|
startWith(string, start) {
|
|
if (!string) return false
|
|
return string.substr(0, start.length) === start
|
|
}
|
|
|
|
withContent() {
|
|
return !!(
|
|
this.consumer().sourcesContent &&
|
|
this.consumer().sourcesContent.length > 0
|
|
)
|
|
}
|
|
}
|
|
|
|
module.exports = PreviousMap
|
|
PreviousMap.default = PreviousMap
|