Files
zellyy-finance/android/app/build.gradle
hansoo 8343b25439 feat: Stage 2 TypeScript 타입 안전성 개선 - any 타입 83개 → 62개 대폭 감소
 주요 개선사항:
- any 타입 83개에서 62개로 21개 수정 (25% 감소)
- 모든 ESLint 에러 11개 → 0개 완전 해결
- 타입 안전성 대폭 향상으로 런타임 오류 가능성 감소

🔧 수정된 파일들:
• PWADebug.tsx - 사용하지 않는 import들에 _ prefix 추가
• categoryUtils.ts - 불필요한 any 캐스트 제거
• TransactionsHeader.tsx - BudgetData 인터페이스 정의
• storageUtils.ts - generic 타입과 unknown 타입 적용
• 각종 error handler들 - Error | {message?: string} 타입 적용
• test 파일들 - 적절한 mock 인터페이스 정의
• 유틸리티 파일들 - any → unknown 또는 적절한 타입으로 교체

🏆 성과:
- 코드 품질 크게 향상 (280 → 80 문제로 71% 감소)
- TypeScript 컴파일러의 타입 체크 효과성 증대
- 개발자 경험 개선 (IDE 자동완성, 타입 추론 등)

🧹 추가 정리:
- ESLint no-console/no-alert 경고 해결
- Prettier 포맷팅 적용으로 코드 스타일 통일

🎯 다음 단계: 남은 62개 any 타입 계속 개선 예정

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-07-14 10:08:51 +09:00

207 lines
9.2 KiB
Groovy

apply plugin: 'com.android.application'
// 버전 정보를 properties 파일에서 동적으로 로드
android {
namespace "com.zellyy.finance"
compileSdkVersion rootProject.ext.compileSdkVersion
defaultConfig {
versionCode 10000
versionName "1.0.0"
applicationId "com.zellyy.finance"
minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion
// 기본 버전 정보 (최신 버전으로 설정)
def defaultVersionName = '1.1.1.3'
def defaultVersionCode = 7
def defaultBuildNumber = 7
// version.properties 파일 로드
def versionPropsFile = file("${rootDir}/version.properties")
def versionProps = new Properties()
try {
if (versionPropsFile.exists() && versionPropsFile.canRead()) {
versionProps.load(new FileInputStream(versionPropsFile))
println "버전 정보 로드: versionName=${versionProps['versionName']}, versionCode=${versionProps['versionCode']}, buildNumber=${versionProps['buildNumber']}"
} else {
println "version.properties 파일을 읽을 수 없음, 기본값 사용: versionName=${defaultVersionName}, versionCode=${defaultVersionCode}, buildNumber=${defaultBuildNumber}"
versionProps['versionName'] = defaultVersionName
versionProps['versionCode'] = defaultVersionCode.toString()
versionProps['buildNumber'] = defaultBuildNumber.toString()
// properties 파일이 없는 경우 생성
try {
versionPropsFile.getParentFile().mkdirs()
versionPropsFile.createNewFile()
def outputStream = new FileOutputStream(versionPropsFile)
versionProps.store(outputStream, "자동 생성된 버전 정보")
outputStream.close()
println "version.properties 파일 생성 성공"
} catch (Exception e) {
println "version.properties 파일 생성 실패: ${e.message}"
}
}
} catch (Exception e) {
println "버전 정보 로드 오류, 기본값 사용: ${e.message}"
versionProps['versionName'] = defaultVersionName
versionProps['versionCode'] = defaultVersionCode.toString()
versionProps['buildNumber'] = defaultBuildNumber.toString()
}
// 로드된 값이 유효한지 확인하고, 유효하지 않으면 기본값 사용
def versionName = defaultVersionName
def versionCode = 9
def buildNumber = defaultBuildNumber
if (versionProps['versionName'] && !versionProps['versionName'].toString().trim().isEmpty()) {
versionName = versionProps['versionName']
} else {
println "유효하지 않은 versionName, 기본값 사용: ${defaultVersionName}"
}
if (versionProps['versionCode'] && !versionProps['versionCode'].toString().trim().isEmpty()) {
try {
// 버전 코드는 고정값 8을 사용하므로 properties에서 읽지 않음
/*
versionCode = versionProps['versionCode'].toString().toInteger()
if (versionCode <= 0) {
println "유효하지 않은 versionCode(0 이하), 기본값 사용: ${defaultVersionCode}"
versionCode = defaultVersionCode
}
*/
} catch (Exception e) {
println "versionCode 변환 오류, 기본값 사용: ${e.message}"
// versionCode = defaultVersionCode
}
} else {
println "유효하지 않은 versionCode, 기본값 사용: ${defaultVersionCode}"
}
if (versionProps['buildNumber'] && !versionProps['buildNumber'].toString().trim().isEmpty()) {
try {
buildNumber = versionProps['buildNumber'].toString().toInteger()
if (buildNumber <= 0) {
println "유효하지 않은 buildNumber(0 이하), 기본값 사용: ${defaultBuildNumber}"
buildNumber = defaultBuildNumber
}
} catch (Exception e) {
println "buildNumber 변환 오류, 기본값 사용: ${e.message}"
buildNumber = defaultBuildNumber
}
} else {
println "유효하지 않은 buildNumber, 기본값 사용: ${defaultBuildNumber}"
}
// 최종 로그 출력
println "최종 버전 정보: versionName=${versionName}, versionCode=${versionCode}, buildNumber=${buildNumber}"
versionCode = 8
// 앱 빌드 속성 설정
// 버전 코드는 이미 고정값 8로 설정됨
// versionCode = versionCode.toInteger()
versionName = "${versionName}"
// 이 부분이 중요합니다 - 이 속성들이 안드로이드 매니페스트에 자동으로 병합됩니다
// BuildConfig 필드 설정
buildConfigField "String", "VERSION_NAME", "\"${versionName}\""
buildConfigField "int", "VERSION_CODE", "${versionCode}"
buildConfigField "int", "BUILD_NUMBER", "${buildNumber}"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
aaptOptions {
// Files and dirs to omit from the packaged assets dir, modified to accommodate modern web apps.
// Default: https://android.googlesource.com/platform/frameworks/base/+/282e181b58cf72b6ca770dc7ca5f91f135444502/tools/aapt/AaptAssets.cpp#61
ignoreAssetsPattern '!.svn:!.git:!.ds_store:!*.scc:.*:!CVS:!thumbs.db:!picasa.ini:!*~'
}
}
// 서명 설정 - CI/CD 및 로컬 개발 환경 지원
signingConfigs {
release {
// CI 환경에서는 환경 변수 사용, 로컬에서는 key.properties 파일 사용
def keystorePropertiesFile = rootProject.file("key.properties")
if (keystorePropertiesFile.exists()) {
// 로컬 개발 환경 - key.properties 파일 사용
def keystoreProperties = new Properties()
keystoreProperties.load(new FileInputStream(keystorePropertiesFile))
storeFile file(keystoreProperties['storeFile'])
storePassword keystoreProperties['storePassword']
keyAlias keystoreProperties['keyAlias']
keyPassword keystoreProperties['keyPassword']
println "로컬 키스토어 설정 로드됨: ${keystoreProperties['storeFile']}"
} else if (System.getenv('CI') == 'true') {
// CI 환경 - 환경 변수 사용
storeFile file('keystore/release.keystore')
storePassword System.getenv('ANDROID_KEYSTORE_PASSWORD')
keyAlias System.getenv('ANDROID_KEY_ALIAS')
keyPassword System.getenv('ANDROID_KEY_PASSWORD')
println "CI 환경 키스토어 설정 로드됨"
} else {
println "⚠️ 키스토어 설정이 없습니다. 릴리즈 빌드가 실패할 수 있습니다."
println "로컬 개발: android/key.properties 파일을 생성하세요."
println "CI 환경: 환경 변수를 설정하세요."
}
}
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
signingConfig signingConfigs.release
}
debug {
minifyEnabled false
debuggable true
}
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_17
targetCompatibility JavaVersion.VERSION_17
}
buildFeatures {
buildConfig true
}
}
repositories {
flatDir{
dirs '../capacitor-cordova-android-plugins/src/main/libs', 'libs'
}
}
dependencies {
implementation fileTree(include: ['*.jar'], dir: 'libs')
implementation "androidx.appcompat:appcompat:$androidxAppCompatVersion"
implementation "androidx.coordinatorlayout:coordinatorlayout:$androidxCoordinatorLayoutVersion"
implementation "androidx.core:core-splashscreen:$coreSplashScreenVersion"
implementation project(':capacitor-android')
testImplementation "junit:junit:$junitVersion"
androidTestImplementation "androidx.test.ext:junit:$androidxJunitVersion"
androidTestImplementation "androidx.test.espresso:espresso-core:$androidxEspressoCoreVersion"
implementation project(':capacitor-cordova-android-plugins')
// 스플래시 화면 플러그인 추가
implementation "androidx.core:core-splashscreen:1.0.0"
}
apply from: 'capacitor.build.gradle'
try {
def servicesJSON = file('google-services.json')
if (servicesJSON.text) {
apply plugin: 'com.google.gms.google-services'
}
} catch(Exception e) {
logger.info("google-services.json not found, google-services plugin not applied. Push Notifications won't work")
}