[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>
64 lines
2.6 KiB
JavaScript
64 lines
2.6 KiB
JavaScript
'use strict';
|
|
|
|
var callBind = require('../');
|
|
var hasStrictMode = require('has-strict-mode')();
|
|
var forEach = require('for-each');
|
|
var inspect = require('object-inspect');
|
|
var v = require('es-value-fixtures');
|
|
|
|
var test = require('tape');
|
|
|
|
test('callBindBasic', function (t) {
|
|
forEach(v.nonFunctions, function (nonFunction) {
|
|
t['throws'](
|
|
// @ts-expect-error
|
|
function () { callBind([nonFunction]); },
|
|
TypeError,
|
|
inspect(nonFunction) + ' is not a function'
|
|
);
|
|
});
|
|
|
|
var sentinel = { sentinel: true };
|
|
/** @type {<T, A extends number, B extends number>(this: T, a: A, b: B) => [T | undefined, A, B]} */
|
|
var func = function (a, b) {
|
|
// eslint-disable-next-line no-invalid-this
|
|
return [!hasStrictMode && this === global ? undefined : this, a, b];
|
|
};
|
|
t.equal(func.length, 2, 'original function length is 2');
|
|
|
|
/** type {(thisArg: unknown, a: number, b: number) => [unknown, number, number]} */
|
|
var bound = callBind([func]);
|
|
/** type {((a: number, b: number) => [typeof sentinel, typeof a, typeof b])} */
|
|
var boundR = callBind([func, sentinel]);
|
|
/** type {((b: number) => [typeof sentinel, number, typeof b])} */
|
|
var boundArg = callBind([func, sentinel, /** @type {const} */ (1)]);
|
|
|
|
// @ts-expect-error
|
|
t.deepEqual(bound(), [undefined, undefined, undefined], 'bound func with no args');
|
|
|
|
// @ts-expect-error
|
|
t.deepEqual(func(), [undefined, undefined, undefined], 'unbound func with too few args');
|
|
// @ts-expect-error
|
|
t.deepEqual(bound(1, 2), [hasStrictMode ? 1 : Object(1), 2, undefined], 'bound func too few args');
|
|
// @ts-expect-error
|
|
t.deepEqual(boundR(), [sentinel, undefined, undefined], 'bound func with receiver, with too few args');
|
|
// @ts-expect-error
|
|
t.deepEqual(boundArg(), [sentinel, 1, undefined], 'bound func with receiver and arg, with too few args');
|
|
|
|
t.deepEqual(func(1, 2), [undefined, 1, 2], 'unbound func with right args');
|
|
t.deepEqual(bound(1, 2, 3), [hasStrictMode ? 1 : Object(1), 2, 3], 'bound func with right args');
|
|
t.deepEqual(boundR(1, 2), [sentinel, 1, 2], 'bound func with receiver, with right args');
|
|
t.deepEqual(boundArg(2), [sentinel, 1, 2], 'bound func with receiver and arg, with right arg');
|
|
|
|
// @ts-expect-error
|
|
t.deepEqual(func(1, 2, 3), [undefined, 1, 2], 'unbound func with too many args');
|
|
// @ts-expect-error
|
|
t.deepEqual(bound(1, 2, 3, 4), [hasStrictMode ? 1 : Object(1), 2, 3], 'bound func with too many args');
|
|
// @ts-expect-error
|
|
t.deepEqual(boundR(1, 2, 3), [sentinel, 1, 2], 'bound func with receiver, with too many args');
|
|
// @ts-expect-error
|
|
t.deepEqual(boundArg(2, 3), [sentinel, 1, 2], 'bound func with receiver and arg, with too many args');
|
|
|
|
t.end();
|
|
});
|