Compare commits
19 Commits
feature/do
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
90dbb86c94 | ||
|
|
8d7ae422e4 | ||
|
|
2a118a24f7 | ||
|
|
3f81939e80 | ||
|
|
7bcf1c5711 | ||
|
|
600190dd80 | ||
|
|
c4a4bdaec3 | ||
|
|
d23cfb7c7d | ||
|
|
dd7adfb96f | ||
|
|
437b9348ce | ||
|
|
6abd80b473 | ||
|
|
c55f469f97 | ||
|
|
02f40e9cc3 | ||
|
|
c5c3e30e78 | ||
|
|
348b98af95 | ||
|
|
0feda01435 | ||
|
|
d920cd5249 | ||
|
|
5a8320bb78 | ||
|
|
410b2b7b48 |
@@ -13,3 +13,4 @@ OIDC_CLIENT_ID=
|
|||||||
OIDC_CLIENT_SECRET=
|
OIDC_CLIENT_SECRET=
|
||||||
OIDC_REDIRECT_URL=
|
OIDC_REDIRECT_URL=
|
||||||
OIDC_SCOPES=openid profile email
|
OIDC_SCOPES=openid profile email
|
||||||
|
IDP_SERVICE_URL=
|
||||||
|
|||||||
54
.gitea/workflows/deploy-registry.yml
Normal file
54
.gitea/workflows/deploy-registry.yml
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
name: Deploy (registry)
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-and-deploy:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Login to registry
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: ${{ vars.HARBOR_ENDPOINT }}
|
||||||
|
username: ${{ vars.HARBOR_ROBOT_ACCOUNT }}
|
||||||
|
password: ${{ secrets.HARBOR_ROBOT_KEY }}
|
||||||
|
|
||||||
|
- name: Build and push
|
||||||
|
uses: docker/build-push-action@v5
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
push: true
|
||||||
|
tags: |
|
||||||
|
${{ vars.HARBOR_ENDPOINT }}/${{ vars.IMAGE_NAME }}:latest
|
||||||
|
${{ vars.HARBOR_ENDPOINT }}/${{ vars.IMAGE_NAME }}:${{ github.sha }}
|
||||||
|
|
||||||
|
- name: Deploy via SSH
|
||||||
|
uses: appleboy/ssh-action@v1.0.3
|
||||||
|
with:
|
||||||
|
host: ${{ vars.SSH_HOST }}
|
||||||
|
username: ${{ vars.SSH_USER }}
|
||||||
|
key: ${{ secrets.SSH_KEY }}
|
||||||
|
port: ${{ vars.SSH_PORT }}
|
||||||
|
script: |
|
||||||
|
cd ${{ secrets.DEPLOY_PATH }}
|
||||||
|
cat << 'ENVEOF' > .env
|
||||||
|
${{ secrets.DEPLOY_ENV_FILE }}
|
||||||
|
ENVEOF
|
||||||
|
|
||||||
|
# Export variables from .env file
|
||||||
|
set -a
|
||||||
|
source .env
|
||||||
|
set +a
|
||||||
|
|
||||||
|
docker compose --env-file .env pull
|
||||||
|
docker compose --env-file .env up -d
|
||||||
54
.gitea/workflows/deploy.yml
Normal file
54
.gitea/workflows/deploy.yml
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
name: Deploy (build on server)
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Deploy via SSH
|
||||||
|
uses: appleboy/ssh-action@v1.0.3
|
||||||
|
with:
|
||||||
|
host: ${{ vars.SSH_HOST }}
|
||||||
|
username: ${{ vars.SSH_USER }}
|
||||||
|
key: ${{ secrets.SSH_KEY }}
|
||||||
|
port: ${{ vars.SSH_PORT }}
|
||||||
|
script: |
|
||||||
|
set -e
|
||||||
|
cd ${{ vars.DEPLOY_PATH }}
|
||||||
|
# if [ ! -d .git ]; then
|
||||||
|
# git init
|
||||||
|
# git remote add origin ssh://git@127.0.0.1:222/b24014/kngil_home.git
|
||||||
|
# else
|
||||||
|
# git remote set-url origin ssh://git@127.0.0.1:222/b24014/kngil_home.git
|
||||||
|
# fi
|
||||||
|
cat << 'ENVEOF' > .env
|
||||||
|
WEB_HOST_PORT=${{ vars.WEB_HOST_PORT }}
|
||||||
|
DB_HOST=${{ vars.DB_HOST }}
|
||||||
|
DB_PORT=${{ vars.DB_PORT }}
|
||||||
|
DB_HOST_PORT=${{ vars.DB_HOST_PORT }}
|
||||||
|
DB_NAME=${{ vars.DB_NAME }}
|
||||||
|
DB_USER=${{ vars.DB_USER }}
|
||||||
|
DB_PASS=${{ secrets.DB_PASS }}
|
||||||
|
OIDC_ISSUER=${{ vars.OIDC_ISSUER }}
|
||||||
|
OIDC_CLIENT_ID=${{ vars.OIDC_CLIENT_ID }}
|
||||||
|
OIDC_CLIENT_SECRET=${{ secrets.OIDC_CLIENT_SECRET }}
|
||||||
|
OIDC_REDIRECT_URL=${{ vars.OIDC_REDIRECT_URL }}
|
||||||
|
OIDC_SCOPES=${{ vars.OIDC_SCOPES }}
|
||||||
|
IDP_SERVICE_URL=${{ vars.IDP_SERVICE_URL }}
|
||||||
|
ENVEOF
|
||||||
|
|
||||||
|
# Export variables from .env file
|
||||||
|
set -a
|
||||||
|
source .env
|
||||||
|
set +a
|
||||||
|
|
||||||
|
git fetch origin main
|
||||||
|
git checkout -B main origin/main
|
||||||
|
git pull --ff-only
|
||||||
|
|
||||||
|
docker compose --env-file .env up -d --build
|
||||||
23
.htaccess
Normal file
23
.htaccess
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
RewriteEngine On
|
||||||
|
|
||||||
|
# Skip existing files and directories.
|
||||||
|
RewriteCond %{REQUEST_FILENAME} -f [OR]
|
||||||
|
RewriteCond %{REQUEST_FILENAME} -d
|
||||||
|
RewriteRule ^ - [L]
|
||||||
|
|
||||||
|
# Admin UI
|
||||||
|
RewriteRule ^admin/?$ /kngil/skin/adm.php [L]
|
||||||
|
RewriteRule ^admin/company/?$ /kngil/skin/adm_comp.php [L]
|
||||||
|
|
||||||
|
# Admin APIs
|
||||||
|
RewriteRule ^admin/api/super/?$ /kngil/bbs/adm.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/company/?$ /kngil/bbs/adm_comp.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/service/?$ /kngil/bbs/adm_service.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/purchase-history/?$ /kngil/bbs/adm_purch_popup.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/use-history/?$ /kngil/bbs/adm_use_history.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/product/?$ /kngil/bbs/adm_product_popup.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/product/save/?$ /kngil/bbs/adm_product_popup_save.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/product/delete/?$ /kngil/bbs/adm_product_popup_delete.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/faq/?$ /kngil/bbs/adm_faq_popup.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/faq/save/?$ /kngil/bbs/adm_faq_popup_save.php [QSA,L]
|
||||||
|
RewriteRule ^admin/api/faq/delete/?$ /kngil/bbs/adm_faq_popup_delete.php [QSA,L]
|
||||||
11
README.md
11
README.md
@@ -2,15 +2,19 @@
|
|||||||
|
|
||||||
## 빠른 시작
|
## 빠른 시작
|
||||||
```bash
|
```bash
|
||||||
docker compose up --build
|
docker compose up -d --build
|
||||||
```
|
```
|
||||||
- 접속: `http://localhost:8080`
|
- 접속: `http://localhost:8080`
|
||||||
|
```bash
|
||||||
|
docker compose down
|
||||||
|
```
|
||||||
|
|
||||||
## 환경변수
|
## 환경변수
|
||||||
`docker-compose.yml`에서 기본값을 사용하며, 필요 시 `.env`로 덮어쓸 수 있습니다.
|
`docker-compose.yml`에서 기본값을 사용하며, 필요 시 `.env`로 덮어쓸 수 있습니다.
|
||||||
|
|
||||||
- `DB_HOST` (기본값: `db`)
|
- `DB_HOST` (기본값: `db`)
|
||||||
- `DB_PORT` (기본값: `5432`)
|
- `DB_PORT` (기본값: `5432`) - 웹 컨테이너가 DB에 접속할 때 사용하는 포트
|
||||||
|
- `DB_HOST_PORT` (기본값: `5432`) - 외부에서 포트포워딩으로 접속할 때 사용하는 호스트 포트
|
||||||
- `DB_NAME` (기본값: `kngil`)
|
- `DB_NAME` (기본값: `kngil`)
|
||||||
- `DB_USER` (기본값: `postgres`)
|
- `DB_USER` (기본값: `postgres`)
|
||||||
- `DB_PASS` (기본값: `postgres`)
|
- `DB_PASS` (기본값: `postgres`)
|
||||||
@@ -25,6 +29,7 @@ docker compose up --build
|
|||||||
- `OIDC_CLIENT_SECRET`
|
- `OIDC_CLIENT_SECRET`
|
||||||
- `OIDC_REDIRECT_URL`
|
- `OIDC_REDIRECT_URL`
|
||||||
- `OIDC_SCOPES` (예: `openid profile email`)
|
- `OIDC_SCOPES` (예: `openid profile email`)
|
||||||
|
- `IDP_SERVICE_URL` (예: `https://idp.example.com`)
|
||||||
|
|
||||||
## DB 초기화
|
## DB 초기화
|
||||||
- `kngil_DB` 덤프는 **처음 실행 시** 자동으로 로드됩니다.
|
- `kngil_DB` 덤프는 **처음 실행 시** 자동으로 로드됩니다.
|
||||||
@@ -43,5 +48,5 @@ docker compose down -v
|
|||||||
- `kngil/bbs/sales_results.php`는 410 응답으로 비활성 처리되어 있습니다.
|
- `kngil/bbs/sales_results.php`는 410 응답으로 비활성 처리되어 있습니다.
|
||||||
|
|
||||||
## PostgreSQL 이미지 버전
|
## PostgreSQL 이미지 버전
|
||||||
- 기본값은 `postgres:18`입니다.
|
- 기본값은 `postgres:16`입니다.
|
||||||
- 이미지 풀 실패 시 `docker-compose.yml`의 태그를 사용 가능한 버전으로 변경하세요.
|
- 이미지 풀 실패 시 `docker-compose.yml`의 태그를 사용 가능한 버전으로 변경하세요.
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ services:
|
|||||||
web:
|
web:
|
||||||
build: .
|
build: .
|
||||||
ports:
|
ports:
|
||||||
- "8080:80"
|
- "${WEB_HOST_PORT:-8080}:80"
|
||||||
volumes:
|
volumes:
|
||||||
- ./:/var/www/html
|
- ./:/var/www/html
|
||||||
environment:
|
environment:
|
||||||
@@ -16,13 +16,21 @@ services:
|
|||||||
OIDC_CLIENT_SECRET: ${OIDC_CLIENT_SECRET:-}
|
OIDC_CLIENT_SECRET: ${OIDC_CLIENT_SECRET:-}
|
||||||
OIDC_REDIRECT_URL: ${OIDC_REDIRECT_URL:-}
|
OIDC_REDIRECT_URL: ${OIDC_REDIRECT_URL:-}
|
||||||
OIDC_SCOPES: ${OIDC_SCOPES:-}
|
OIDC_SCOPES: ${OIDC_SCOPES:-}
|
||||||
|
IDP_SERVICE_URL: ${IDP_SERVICE_URL:-}
|
||||||
|
restart: always
|
||||||
depends_on:
|
depends_on:
|
||||||
- db
|
- db
|
||||||
|
|
||||||
db:
|
db:
|
||||||
image: postgres:16
|
image: postgres:16
|
||||||
# ports:
|
ports:
|
||||||
# - "5432:5432"
|
- "0.0.0.0:${DB_HOST_PORT:-5432}:5432"
|
||||||
|
command:
|
||||||
|
- "postgres"
|
||||||
|
- "-c"
|
||||||
|
- "listen_addresses=*"
|
||||||
|
- "-c"
|
||||||
|
- "hba_file=/etc/postgresql/pg_hba.conf"
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_DB: ${DB_NAME:-kngil}
|
POSTGRES_DB: ${DB_NAME:-kngil}
|
||||||
POSTGRES_USER: ${DB_USER:-postgres}
|
POSTGRES_USER: ${DB_USER:-postgres}
|
||||||
@@ -30,6 +38,8 @@ services:
|
|||||||
volumes:
|
volumes:
|
||||||
- db_data:/var/lib/postgresql/data
|
- db_data:/var/lib/postgresql/data
|
||||||
- ./docker/initdb/01_kngil_DB.sql:/docker-entrypoint-initdb.d/01_kngil_DB.sql:ro
|
- ./docker/initdb/01_kngil_DB.sql:/docker-entrypoint-initdb.d/01_kngil_DB.sql:ro
|
||||||
|
- ./docker/postgres/pg_hba.conf:/etc/postgresql/pg_hba.conf:ro
|
||||||
|
restart: always
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
db_data:
|
db_data:
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
set -e
|
set -e
|
||||||
|
|
||||||
if [ ! -f /var/www/html/kngil/vendor/autoload.php ]; then
|
if [ ! -f /var/www/html/kngil/vendor/autoload.php ]; then
|
||||||
|
echo "[초기화] composer install을 실행합니다..."
|
||||||
php /var/www/html/kngil/composer.phar install --working-dir=/var/www/html/kngil --no-interaction --prefer-dist
|
php /var/www/html/kngil/composer.phar install --working-dir=/var/www/html/kngil --no-interaction --prefer-dist
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|||||||
12
docker/initdb/00_create_db.sh
Executable file
12
docker/initdb/00_create_db.sh
Executable file
@@ -0,0 +1,12 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
set -e
|
||||||
|
|
||||||
|
psql -v ON_ERROR_STOP=1 --username "$POSTGRES_USER" --dbname "postgres" <<'EOSQL'
|
||||||
|
DO $$
|
||||||
|
BEGIN
|
||||||
|
IF NOT EXISTS (SELECT 1 FROM pg_database WHERE datname = 'kngil') THEN
|
||||||
|
CREATE DATABASE kngil;
|
||||||
|
END IF;
|
||||||
|
END
|
||||||
|
$$;
|
||||||
|
EOSQL
|
||||||
@@ -2378,4 +2378,3 @@ ALTER TABLE ONLY kngil.use_history
|
|||||||
--
|
--
|
||||||
|
|
||||||
\unrestrict osPaC8Gqjay0KBMwX4hwgDvmjwF5rTGmBMzQBdxAne3SBCLMuCNQu2Xg15dPVeb
|
\unrestrict osPaC8Gqjay0KBMwX4hwgDvmjwF5rTGmBMzQBdxAne3SBCLMuCNQu2Xg15dPVeb
|
||||||
|
|
||||||
|
|||||||
11
docker/postgres/pg_hba.conf
Normal file
11
docker/postgres/pg_hba.conf
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
#
|
||||||
|
# 로컬 소켓 접속 허용(초기화/관리용). 운영에서는 정책에 맞게 조정하세요.
|
||||||
|
#
|
||||||
|
local all all trust
|
||||||
|
host all all 127.0.0.1/32 scram-sha-256
|
||||||
|
host all all ::1/128 scram-sha-256
|
||||||
|
#
|
||||||
|
# 외부 TCP 접속 허용. 운영에서는 허용 대역을 제한하세요.
|
||||||
|
#
|
||||||
|
host all all 0.0.0.0/0 scram-sha-256
|
||||||
|
host all all ::/0 scram-sha-256
|
||||||
@@ -7,7 +7,7 @@ declare(strict_types=1);
|
|||||||
// 1. 기본 상수
|
// 1. 기본 상수
|
||||||
// ---------------------------------
|
// ---------------------------------
|
||||||
define('ROOT', __DIR__);
|
define('ROOT', __DIR__);
|
||||||
define('SKIN_PATH', ROOT.'/skin');
|
define('SKIN_PATH', ROOT.'/kngil/skin');
|
||||||
|
|
||||||
// ---------------------------------
|
// ---------------------------------
|
||||||
// 2. 페이지 결정
|
// 2. 페이지 결정
|
||||||
|
|||||||
@@ -1,12 +1,17 @@
|
|||||||
<?php
|
<?php
|
||||||
// /kngil/auth/oidc-callback.php
|
// /kngil/auth/oidc-callback.php
|
||||||
session_start();
|
session_start();
|
||||||
|
ini_set('log_errors', '1');
|
||||||
|
ini_set('error_log', '/proc/self/fd/2');
|
||||||
require_once dirname(__DIR__) . '/vendor/autoload.php';
|
require_once dirname(__DIR__) . '/vendor/autoload.php';
|
||||||
require_once dirname(__DIR__) . '/bbs/db_conn.php';
|
require_once dirname(__DIR__) . '/bbs/db_conn.php';
|
||||||
$config = require_once dirname(__DIR__) . '/bbs/oidc_config.php';
|
$config = require_once dirname(__DIR__) . '/bbs/oidc_config.php';
|
||||||
|
|
||||||
use Jumbojett\OpenIDConnectClient;
|
use Jumbojett\OpenIDConnectClient;
|
||||||
|
|
||||||
|
$usersTable = 'kngil.users';
|
||||||
|
$membersTable = 'kngil.members';
|
||||||
|
|
||||||
$oidc = new OpenIDConnectClient(
|
$oidc = new OpenIDConnectClient(
|
||||||
$config['issuer'],
|
$config['issuer'],
|
||||||
$config['client_id'],
|
$config['client_id'],
|
||||||
@@ -16,16 +21,124 @@ $oidc = new OpenIDConnectClient(
|
|||||||
$oidc->setRedirectURL($config['redirect_url']);
|
$oidc->setRedirectURL($config['redirect_url']);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
$stmt = $pdo->query("SELECT to_regclass('kngil.users') AS reg");
|
||||||
|
$reg = $stmt ? $stmt->fetchColumn() : null;
|
||||||
|
if (!$reg) {
|
||||||
|
$stmt = $pdo->query("SELECT to_regclass('public.users') AS reg");
|
||||||
|
$reg = $stmt ? $stmt->fetchColumn() : null;
|
||||||
|
if ($reg) {
|
||||||
|
$usersTable = 'public.users';
|
||||||
|
$membersTable = 'public.members';
|
||||||
|
} else {
|
||||||
|
throw new Exception(
|
||||||
|
"사용자 테이블을 찾을 수 없습니다. DB 초기화가 필요합니다. "
|
||||||
|
. "docker compose down -v 후 다시 실행하거나, "
|
||||||
|
. "DB_NAME/DB_USER/DB_PASS 설정을 확인하세요."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$memberReg = $pdo->query("SELECT to_regclass('{$membersTable}') AS reg");
|
||||||
|
$memberReg = $memberReg ? $memberReg->fetchColumn() : null;
|
||||||
|
if (!$memberReg) {
|
||||||
|
$altMembersTable = $membersTable === 'kngil.members' ? 'public.members' : 'kngil.members';
|
||||||
|
$memberReg = $pdo->query("SELECT to_regclass('{$altMembersTable}') AS reg");
|
||||||
|
$memberReg = $memberReg ? $memberReg->fetchColumn() : null;
|
||||||
|
if ($memberReg) {
|
||||||
|
$membersTable = $altMembersTable;
|
||||||
|
} else {
|
||||||
|
throw new Exception("회원 테이블을 찾을 수 없습니다. DB 초기화가 필요합니다.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$pdo->exec("ALTER TABLE {$usersTable} ADD COLUMN IF NOT EXISTS oidc_sub VARCHAR(255) UNIQUE");
|
||||||
|
|
||||||
if (!$oidc->authenticate()) {
|
if (!$oidc->authenticate()) {
|
||||||
throw new Exception("Authentication failed");
|
throw new Exception("Authentication failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
$userInfo = $oidc->requestUserInfo();
|
$userInfo = $oidc->requestUserInfo();
|
||||||
|
$idToken = $oidc->getIdToken();
|
||||||
|
$accessToken = $oidc->getAccessToken();
|
||||||
|
$jwtClaims = [];
|
||||||
|
if (!empty($idToken)) {
|
||||||
|
$parts = explode('.', $idToken);
|
||||||
|
if (count($parts) >= 2) {
|
||||||
|
$payload = strtr($parts[1], '-_', '+/');
|
||||||
|
$padding = 4 - (strlen($payload) % 4);
|
||||||
|
if ($padding < 4) {
|
||||||
|
$payload .= str_repeat('=', $padding);
|
||||||
|
}
|
||||||
|
$decoded = json_decode(base64_decode($payload), true);
|
||||||
|
if (is_array($decoded)) {
|
||||||
|
$jwtClaims = $decoded;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 디버그용: ID 토큰 확보 여부 로그 출력 (파일)
|
||||||
|
$logDir = dirname(__DIR__) . '/log';
|
||||||
|
if (!is_dir($logDir)) {
|
||||||
|
@mkdir($logDir, 0775, true);
|
||||||
|
}
|
||||||
|
$logPath = $logDir . '/oidc_debug.log';
|
||||||
|
if (!is_writable($logDir)) {
|
||||||
|
$logPath = '/tmp/oidc_debug.log';
|
||||||
|
error_log('[OIDC_DEBUG] log_dir_not_writable, fallback=/tmp/oidc_debug.log');
|
||||||
|
}
|
||||||
|
$tokenInfo = empty($idToken) ? 'MISSING' : ('PRESENT len=' . strlen($idToken));
|
||||||
|
$claimKeys = empty($jwtClaims) ? 'none' : implode(',', array_keys($jwtClaims));
|
||||||
|
$logLine = sprintf(
|
||||||
|
"[%s] host=%s uri=%s sid=%s id_token=%s claims=%s\n",
|
||||||
|
date('c'),
|
||||||
|
$_SERVER['HTTP_HOST'] ?? '-',
|
||||||
|
$_SERVER['REQUEST_URI'] ?? '-',
|
||||||
|
session_id(),
|
||||||
|
$tokenInfo,
|
||||||
|
$claimKeys
|
||||||
|
);
|
||||||
|
$writeOk = @file_put_contents($logPath, $logLine, FILE_APPEND);
|
||||||
|
if ($writeOk === false) {
|
||||||
|
error_log('[OIDC_DEBUG] log_write_failed path=' . $logPath);
|
||||||
|
}
|
||||||
|
// 디버그용: userInfo/claims 전체 덤프 (토큰 제외)
|
||||||
|
$dump = [
|
||||||
|
'userInfo' => $userInfo,
|
||||||
|
'jwtClaims' => $jwtClaims
|
||||||
|
];
|
||||||
|
$dumpLine = sprintf(
|
||||||
|
"[%s] oidc_dump=%s\n",
|
||||||
|
date('c'),
|
||||||
|
json_encode($dump, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES)
|
||||||
|
);
|
||||||
|
$dumpOk = @file_put_contents($logPath, $dumpLine, FILE_APPEND);
|
||||||
|
if ($dumpOk === false) {
|
||||||
|
error_log('[OIDC_DEBUG] dump_write_failed path=' . $logPath);
|
||||||
|
}
|
||||||
|
// 도커 로그로도 출력
|
||||||
|
error_log('[OIDC_DEBUG] ' . $dumpLine);
|
||||||
// $userInfo 에 포함된 데이터 예시: sub, email, name, preferred_username 등
|
// $userInfo 에 포함된 데이터 예시: sub, email, name, preferred_username 등
|
||||||
|
|
||||||
$email = $userInfo->email ?? null;
|
$email = $userInfo->email ?? null;
|
||||||
$sub = $userInfo->sub ?? null; // IDP 고유 식별자
|
$sub = $userInfo->sub ?? null; // IDP 고유 식별자
|
||||||
$name = $userInfo->name ?? ($userInfo->preferred_username ?? 'Unknown');
|
$preferred = $userInfo->preferred_username ?? null;
|
||||||
|
$name = $userInfo->name ?? null;
|
||||||
|
if (!$email && isset($jwtClaims['email'])) {
|
||||||
|
$email = $jwtClaims['email'];
|
||||||
|
}
|
||||||
|
if (!$name && isset($jwtClaims['name'])) {
|
||||||
|
$name = $jwtClaims['name'];
|
||||||
|
}
|
||||||
|
if (!$name && $preferred) {
|
||||||
|
$name = $preferred;
|
||||||
|
}
|
||||||
|
if (!$name && $email) {
|
||||||
|
$name = $email;
|
||||||
|
}
|
||||||
|
if (!$name && $sub) {
|
||||||
|
$seed = strtolower(preg_replace('/[^a-z0-9]/', '', (string)$sub));
|
||||||
|
$name = 'oidc_' . substr($seed, 0, 10);
|
||||||
|
}
|
||||||
|
|
||||||
if (!$email && !$sub) {
|
if (!$email && !$sub) {
|
||||||
throw new Exception("IDP provided insufficient user information.");
|
throw new Exception("IDP provided insufficient user information.");
|
||||||
@@ -33,7 +146,7 @@ try {
|
|||||||
|
|
||||||
// 1. 사용자 매핑 (sub 또는 email 기준)
|
// 1. 사용자 매핑 (sub 또는 email 기준)
|
||||||
$stmt = $pdo->prepare("
|
$stmt = $pdo->prepare("
|
||||||
SELECT * FROM kngil.users
|
SELECT * FROM {$usersTable}
|
||||||
WHERE (oidc_sub = :sub OR LOWER(email) = LOWER(:email))
|
WHERE (oidc_sub = :sub OR LOWER(email) = LOWER(:email))
|
||||||
AND use_yn = 'Y'
|
AND use_yn = 'Y'
|
||||||
LIMIT 1
|
LIMIT 1
|
||||||
@@ -42,14 +155,99 @@ try {
|
|||||||
$user = $stmt->fetch(PDO::FETCH_ASSOC);
|
$user = $stmt->fetch(PDO::FETCH_ASSOC);
|
||||||
|
|
||||||
if (!$user) {
|
if (!$user) {
|
||||||
// [정책 선택] 새 사용자 자동 생성 또는 로그인 거부
|
$defaultMemberId = getenv('OIDC_DEFAULT_MEMBER_ID') ?: '';
|
||||||
// 여기서는 예시로 로그인 거부 처리
|
if ($defaultMemberId !== '') {
|
||||||
throw new Exception("등록되지 않은 사용자입니다. 관리자에게 문의하세요. (IDP: $email)");
|
$checkMember = $pdo->prepare("SELECT 1 FROM {$membersTable} WHERE member_id = :member_id LIMIT 1");
|
||||||
|
$checkMember->execute([':member_id' => $defaultMemberId]);
|
||||||
|
if (!$checkMember->fetchColumn()) {
|
||||||
|
throw new Exception("OIDC_DEFAULT_MEMBER_ID가 members에 존재하지 않습니다: {$defaultMemberId}");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$memberStmt = $pdo->query("SELECT member_id FROM {$membersTable} ORDER BY member_id ASC LIMIT 1");
|
||||||
|
$defaultMemberId = $memberStmt ? $memberStmt->fetchColumn() : '';
|
||||||
|
if (!$defaultMemberId) {
|
||||||
|
throw new Exception("기본 member_id를 찾을 수 없습니다. OIDC_DEFAULT_MEMBER_ID를 설정하세요.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$defaultAuth = getenv('OIDC_DEFAULT_AUTH_BC') ?: 'BS100500';
|
||||||
|
|
||||||
|
$baseId = $userInfo->preferred_username ?? ($email ? explode('@', $email)[0] : '');
|
||||||
|
$baseId = strtolower(preg_replace('/[^a-z0-9]/', '', $baseId));
|
||||||
|
if ($baseId === '') {
|
||||||
|
$seed = strtolower(preg_replace('/[^a-z0-9]/', '', (string)($sub ?? 'oidc')));
|
||||||
|
$baseId = 'oidc' . substr($seed, 0, 10);
|
||||||
|
}
|
||||||
|
$baseId = substr($baseId, 0, 16);
|
||||||
|
$userId = $baseId;
|
||||||
|
|
||||||
|
$existsStmt = $pdo->prepare("SELECT 1 FROM {$usersTable} WHERE LOWER(user_id) = LOWER(:user_id) LIMIT 1");
|
||||||
|
$suffix = 1;
|
||||||
|
while (true) {
|
||||||
|
$existsStmt->execute([':user_id' => $userId]);
|
||||||
|
if (!$existsStmt->fetchColumn()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$tail = sprintf('%02d', $suffix);
|
||||||
|
$userId = substr($baseId, 0, 20 - strlen($tail)) . $tail;
|
||||||
|
$suffix++;
|
||||||
|
if ($suffix > 99) {
|
||||||
|
$userId = 'oidc' . bin2hex(random_bytes(4));
|
||||||
|
$userId = substr($userId, 0, 20);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$userNm = $name ?: ($email ?: $userId);
|
||||||
|
$rawPhone = $userInfo->phone_number ?? '';
|
||||||
|
$digits = preg_replace('/\D/', '', $rawPhone);
|
||||||
|
if (strlen($digits) === 11) {
|
||||||
|
$telNo = substr($digits, 0, 3) . '-' . substr($digits, 3, 4) . '-' . substr($digits, 7, 4);
|
||||||
|
} elseif (strlen($digits) === 10) {
|
||||||
|
$telNo = substr($digits, 0, 3) . '-' . substr($digits, 3, 3) . '-' . substr($digits, 6, 4);
|
||||||
|
} else {
|
||||||
|
$telNo = '000-0000-0000';
|
||||||
|
}
|
||||||
|
|
||||||
|
$insert = $pdo->prepare("
|
||||||
|
INSERT INTO {$usersTable} (
|
||||||
|
member_id, user_id, user_pw, user_nm,
|
||||||
|
dept_nm, posit_nm, tel_no, email,
|
||||||
|
auth_bc, use_yn, rmks,
|
||||||
|
cid, cdt, mid, mdt, oidc_sub
|
||||||
|
) VALUES (
|
||||||
|
:member_id, :user_id, NULL, :user_nm,
|
||||||
|
:dept_nm, :posit_nm, :tel_no, :email,
|
||||||
|
:auth_bc, 'Y', :rmks,
|
||||||
|
:cid, CURRENT_TIMESTAMP, :mid, CURRENT_TIMESTAMP, :oidc_sub
|
||||||
|
)
|
||||||
|
");
|
||||||
|
$insert->execute([
|
||||||
|
':member_id' => $defaultMemberId,
|
||||||
|
':user_id' => $userId,
|
||||||
|
':user_nm' => $userNm,
|
||||||
|
':dept_nm' => $userInfo->department ?? null,
|
||||||
|
':posit_nm' => $userInfo->title ?? null,
|
||||||
|
':tel_no' => $telNo,
|
||||||
|
':email' => $email,
|
||||||
|
':auth_bc' => $defaultAuth,
|
||||||
|
':rmks' => 'OIDC auto-registered',
|
||||||
|
':cid' => $userId,
|
||||||
|
':mid' => $userId,
|
||||||
|
':oidc_sub' => $sub
|
||||||
|
]);
|
||||||
|
|
||||||
|
$stmt = $pdo->prepare("
|
||||||
|
SELECT * FROM {$usersTable}
|
||||||
|
WHERE LOWER(user_id) = LOWER(:user_id)
|
||||||
|
LIMIT 1
|
||||||
|
");
|
||||||
|
$stmt->execute([':user_id' => $userId]);
|
||||||
|
$user = $stmt->fetch(PDO::FETCH_ASSOC);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. oidc_sub 업데이트 (최초 연동 시)
|
// 2. oidc_sub 업데이트 (최초 연동 시)
|
||||||
if (empty($user['oidc_sub']) && $sub) {
|
if (empty($user['oidc_sub']) && $sub) {
|
||||||
$upd = $pdo->prepare("UPDATE kngil.users SET oidc_sub = :sub WHERE user_id = :id");
|
$upd = $pdo->prepare("UPDATE {$usersTable} SET oidc_sub = :sub WHERE user_id = :id");
|
||||||
$upd->execute([':sub' => $sub, ':id' => $user['user_id']]);
|
$upd->execute([':sub' => $sub, ':id' => $user['user_id']]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,20 +261,42 @@ try {
|
|||||||
'dept_nm' => $user['dept_nm'] ?? null,
|
'dept_nm' => $user['dept_nm'] ?? null,
|
||||||
'tel_no' => $user['tel_no'] ?? null,
|
'tel_no' => $user['tel_no'] ?? null,
|
||||||
'email' => $user['email'] ?? null,
|
'email' => $user['email'] ?? null,
|
||||||
|
'idp_name' => $name ?: null,
|
||||||
|
'idp_email' => $email ?? null,
|
||||||
|
'idp_id_token' => $idToken ?? null,
|
||||||
|
'idp_access_token' => $accessToken ?? null,
|
||||||
|
'idp_claims' => $jwtClaims ?? null,
|
||||||
'oidc_mode' => true // OIDC 로그인을 나타내는 플래그
|
'oidc_mode' => true // OIDC 로그인을 나타내는 플래그
|
||||||
];
|
];
|
||||||
|
|
||||||
// 로그인 완료 후 부모 창에 알리고 종료
|
session_write_close();
|
||||||
|
|
||||||
|
// 로그인 완료 후 부모 창에 알리고 종료 (팝업이 아닐 경우 메인으로 이동)
|
||||||
?>
|
?>
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html>
|
<html>
|
||||||
<body>
|
<body>
|
||||||
<script>
|
<script>
|
||||||
if (window.opener) {
|
(function () {
|
||||||
window.opener.postMessage({ type: 'OIDC_LOGIN_SUCCESS' }, window.location.origin);
|
const target = '/kngil/skin/index.php';
|
||||||
}
|
if (window.opener && !window.opener.closed) {
|
||||||
window.close();
|
try {
|
||||||
|
window.opener.postMessage({ type: 'OIDC_LOGIN_SUCCESS' }, window.location.origin);
|
||||||
|
} catch (e) {
|
||||||
|
// 팝업 차단/보안 정책으로 실패할 수 있어 무시합니다.
|
||||||
|
}
|
||||||
|
window.close();
|
||||||
|
setTimeout(function () {
|
||||||
|
window.location.href = target;
|
||||||
|
}, 300);
|
||||||
|
} else {
|
||||||
|
window.location.href = target;
|
||||||
|
}
|
||||||
|
})();
|
||||||
</script>
|
</script>
|
||||||
|
<noscript>
|
||||||
|
<a href="/kngil/skin/index.php">메인으로 이동</a>
|
||||||
|
</noscript>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
<?php
|
<?php
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ if (session_status() === PHP_SESSION_NONE) {
|
|||||||
1. 로그인 체크
|
1. 로그인 체크
|
||||||
========================= */
|
========================= */
|
||||||
if (empty($_SESSION['login'])) {
|
if (empty($_SESSION['login'])) {
|
||||||
header('Location: /kngil/skin/index.php');
|
header('Location: /');
|
||||||
exit;
|
exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -33,4 +33,4 @@ if (!in_array($auth_bc, $ALLOW_AUTH, true)) {
|
|||||||
3. 권한 플래그 (중요)
|
3. 권한 플래그 (중요)
|
||||||
========================= */
|
========================= */
|
||||||
define('IS_SUPER_ADMIN', in_array($auth_bc, ['BS100100','BS100200'], true));
|
define('IS_SUPER_ADMIN', in_array($auth_bc, ['BS100100','BS100200'], true));
|
||||||
define('IS_COMPANY_ADMIN', in_array($auth_bc, ['BS100300','BS100400'], true));
|
define('IS_COMPANY_ADMIN', in_array($auth_bc, ['BS100300','BS100400'], true));
|
||||||
|
|||||||
@@ -1,5 +1,9 @@
|
|||||||
@charset "UTF-8";
|
@charset "UTF-8";
|
||||||
@import url("https://fonts.googleapis.com/css2?family=Noto+Sans+KR:wght@100;300;400;500;700;900&display=swap");
|
@import url("https://fonts.googleapis.com/css2?family=Noto+Sans+KR:wght@100;300;400;500;700;900&display=swap");
|
||||||
|
:root {
|
||||||
|
--color-primary: #f95523;
|
||||||
|
--color-primary-border: #ca3f14;
|
||||||
|
}
|
||||||
html {
|
html {
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
}
|
}
|
||||||
@@ -3697,4 +3701,4 @@ i {
|
|||||||
html {
|
html {
|
||||||
font-size: 8px;
|
font-size: 8px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -102,7 +102,7 @@ userGrid
|
|||||||
document.getElementById('detailCard').style.display = 'block'
|
document.getElementById('detailCard').style.display = 'block'
|
||||||
|
|
||||||
// 하단 사용자 로드
|
// 하단 사용자 로드
|
||||||
// fetch(`/kngil/bbs/adm.php?action=user_list&member_id=${record.member_id}`)
|
// fetch(`/admin/api/super?action=user_list&member_id=${record.member_id}`)
|
||||||
// .then(res => res.json())
|
// .then(res => res.json())
|
||||||
// .then(d => {
|
// .then(d => {
|
||||||
// if (d.status !== 'success') {
|
// if (d.status !== 'success') {
|
||||||
@@ -243,7 +243,7 @@ function formatBizNo(value) {
|
|||||||
상단 회사 목록 로드
|
상단 회사 목록 로드
|
||||||
---------------------------------------- */
|
---------------------------------------- */
|
||||||
function loadCompanies() {
|
function loadCompanies() {
|
||||||
fetch('/kngil/bbs/adm.php')
|
fetch('/admin/api/super')
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
if (!json.records) return
|
if (!json.records) return
|
||||||
@@ -340,7 +340,7 @@ export function bindSaveButton() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm.php?action=save', {
|
fetch('/admin/api/super?action=save', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ function destroyGrid(name) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function loadBaseCode(mainCd) {
|
function loadBaseCode(mainCd) {
|
||||||
return fetch(`/kngil/bbs/adm_comp.php?action=base_code&main_cd=${mainCd}`)
|
return fetch(`/admin/api/company?action=base_code&main_cd=${mainCd}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
if (json.status !== 'success') {
|
if (json.status !== 'success') {
|
||||||
@@ -144,7 +144,7 @@ export async function createUserGrid(boxId, options = {}) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function loadUsers() {
|
function loadUsers() {
|
||||||
fetch('/kngil/bbs/adm_comp.php')
|
fetch('/admin/api/company')
|
||||||
.then(res => res.text()) // 🔥 먼저 text로 확인
|
.then(res => res.text()) // 🔥 먼저 text로 확인
|
||||||
.then(text => {
|
.then(text => {
|
||||||
try {
|
try {
|
||||||
@@ -168,7 +168,7 @@ export function loadUsersByMember(member_id) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_comp.php')
|
fetch('/admin/api/company')
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
g.clear()
|
g.clear()
|
||||||
@@ -198,7 +198,7 @@ export function setUserGridMode(mode = 'view') {
|
|||||||
|
|
||||||
export function loadData({ loadSummary = true } = {}) {
|
export function loadData({ loadSummary = true } = {}) {
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_comp.php')
|
fetch('/admin/api/company')
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(async d => {
|
.then(async d => {
|
||||||
|
|
||||||
@@ -323,7 +323,7 @@ document.getElementById('btnSave_comp')?.addEventListener('click', () => {
|
|||||||
console.log('INSERTS', inserts)
|
console.log('INSERTS', inserts)
|
||||||
console.log('UPDATES', updates)
|
console.log('UPDATES', updates)
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_comp.php', {
|
fetch('/admin/api/company', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -408,7 +408,7 @@ document.getElementById('btnDelete')?.addEventListener('click', () => {
|
|||||||
w2confirm(`선택한 ${ids.length}명의 사용자를 삭제하시겠습니까?`)
|
w2confirm(`선택한 ${ids.length}명의 사용자를 삭제하시겠습니까?`)
|
||||||
.yes(() => {
|
.yes(() => {
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_comp.php', {
|
fetch('/admin/api/company', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -436,7 +436,7 @@ document.getElementById('btnDelete')?.addEventListener('click', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
function loadTotalArea(memberId) {
|
function loadTotalArea(memberId) {
|
||||||
return fetch(`/kngil/bbs/adm_comp.php?action=total_area&member_id=${memberId}`)
|
return fetch(`/admin/api/company?action=total_area&member_id=${memberId}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
if (json.status !== 'success') {
|
if (json.status !== 'success') {
|
||||||
@@ -467,7 +467,7 @@ function doSearch() {
|
|||||||
}
|
}
|
||||||
// ⚠️ type === 'id' 는 DB로 안 보냄
|
// ⚠️ type === 'id' 는 DB로 안 보냄
|
||||||
|
|
||||||
fetch(`/kngil/bbs/adm_comp.php?action=list`
|
fetch(`/admin/api/company?action=list`
|
||||||
+ `&user_nm=${encodeURIComponent(p_user_nm)}`
|
+ `&user_nm=${encodeURIComponent(p_user_nm)}`
|
||||||
+ `&dept_nm=${encodeURIComponent(p_dept_nm)}`
|
+ `&dept_nm=${encodeURIComponent(p_dept_nm)}`
|
||||||
+ `&use_yn=${useYn}`
|
+ `&use_yn=${useYn}`
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ function destroyGrid(name) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function loadBaseCode(mainCd) {
|
function loadBaseCode(mainCd) {
|
||||||
return fetch(`/kngil/bbs/adm_comp.php?action=base_code&main_cd=${mainCd}`)
|
return fetch(`/admin/api/company?action=base_code&main_cd=${mainCd}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
if (json.status !== 'success') {
|
if (json.status !== 'success') {
|
||||||
@@ -251,7 +251,7 @@ export function loadUsersByMember(memberId) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
fetch(`/kngil/bbs/adm_comp.php?action=list&member_id=${memberId}`)
|
fetch(`/admin/api/company?action=list&member_id=${memberId}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(d => {
|
.then(d => {
|
||||||
|
|
||||||
@@ -285,7 +285,7 @@ export function setUserGridMode(mode = 'view') {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function loadData({ loadSummary = true } = {}) {
|
export function loadData({ loadSummary = true } = {}) {
|
||||||
fetch('/kngil/bbs/adm_comp.php?action=list')
|
fetch('/admin/api/company?action=list')
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(async d => {
|
.then(async d => {
|
||||||
|
|
||||||
@@ -404,7 +404,7 @@ document.getElementById('btnSave_comp')?.addEventListener('click', () => {
|
|||||||
console.log('INSERTS', inserts)
|
console.log('INSERTS', inserts)
|
||||||
console.log('UPDATES', updates)
|
console.log('UPDATES', updates)
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_comp.php', {
|
fetch('/admin/api/company', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -489,7 +489,7 @@ document.getElementById('btnDelete')?.addEventListener('click', () => {
|
|||||||
w2confirm(`선택한 ${ids.length}명의 사용자를 삭제하시겠습니까?`)
|
w2confirm(`선택한 ${ids.length}명의 사용자를 삭제하시겠습니까?`)
|
||||||
.yes(() => {
|
.yes(() => {
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_comp.php', {
|
fetch('/admin/api/company', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -517,7 +517,7 @@ document.getElementById('btnDelete')?.addEventListener('click', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
function loadTotalArea(memberId) {
|
function loadTotalArea(memberId) {
|
||||||
return fetch(`/kngil/bbs/adm_comp.php?action=total_area`)
|
return fetch(`/admin/api/company?action=total_area`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
if (json.status !== 'success') {
|
if (json.status !== 'success') {
|
||||||
@@ -548,7 +548,7 @@ function doSearch() {
|
|||||||
}
|
}
|
||||||
// ⚠️ type === 'id' 는 DB로 안 보냄
|
// ⚠️ type === 'id' 는 DB로 안 보냄
|
||||||
|
|
||||||
fetch(`/kngil/bbs/adm_comp.php?action=list`
|
fetch(`/admin/api/company?action=list`
|
||||||
+ `&user_nm=${encodeURIComponent(p_user_nm)}`
|
+ `&user_nm=${encodeURIComponent(p_user_nm)}`
|
||||||
+ `&dept_nm=${encodeURIComponent(p_dept_nm)}`
|
+ `&dept_nm=${encodeURIComponent(p_dept_nm)}`
|
||||||
+ `&use_yn=${useYn}`
|
+ `&use_yn=${useYn}`
|
||||||
@@ -670,7 +670,7 @@ function openBulkCreatePopup(memberId) {
|
|||||||
|
|
||||||
function runBulkCreate(memberId, csvUrl) {
|
function runBulkCreate(memberId, csvUrl) {
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_comp.php', {
|
fetch('/admin/api/company', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -771,7 +771,7 @@ function loadDataByMemberId(memberId) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
fetch(`/kngil/bbs/adm_comp.php?action=list&member_id=${encodeURIComponent(memberId)}`)
|
fetch(`/admin/api/company?action=list&member_id=${encodeURIComponent(memberId)}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(async d => {
|
.then(async d => {
|
||||||
|
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ function destroyGrid(name) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function loadBaseCode(mainCd) {
|
function loadBaseCode(mainCd) {
|
||||||
return fetch(`/kngil/bbs/adm_comp.php?action=base_code&main_cd=${mainCd}`)
|
return fetch(`/admin/api/company?action=base_code&main_cd=${mainCd}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
if (json.status !== 'success') {
|
if (json.status !== 'success') {
|
||||||
@@ -100,7 +100,7 @@ export function openfaqPopup() {
|
|||||||
|
|
||||||
// 3. 브라우저 기본 확인창 사용 (가장 확실함)
|
// 3. 브라우저 기본 확인창 사용 (가장 확실함)
|
||||||
if (confirm(`선택한 ${ids.length}개의 상품을 삭제하시겠습니까?`)) {
|
if (confirm(`선택한 ${ids.length}개의 상품을 삭제하시겠습니까?`)) {
|
||||||
fetch('/kngil/bbs/adm_faq_popup_delete.php', {
|
fetch('/admin/api/faq/delete', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({ action: 'delete', ids: ids })
|
body: JSON.stringify({ action: 'delete', ids: ids })
|
||||||
@@ -169,7 +169,7 @@ export function openfaqPopup() {
|
|||||||
console.log('INSERTS', inserts)
|
console.log('INSERTS', inserts)
|
||||||
console.log('UPDATES', updates)
|
console.log('UPDATES', updates)
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_faq_popup_save.php', {
|
fetch('/admin/api/faq/save', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -270,7 +270,7 @@ async function loadfaqData() {
|
|||||||
try {
|
try {
|
||||||
w2ui.faqGrid.lock('조회 중...', true);
|
w2ui.faqGrid.lock('조회 중...', true);
|
||||||
|
|
||||||
const response = await fetch('/kngil/bbs/adm_faq_popup.php'); // PHP 파일 호출
|
const response = await fetch('/admin/api/faq'); // PHP 파일 호출
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
|
|
||||||
w2ui.faqGrid.clear();
|
w2ui.faqGrid.clear();
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ function destroyGrid(name) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function loadBaseCode(mainCd) {
|
function loadBaseCode(mainCd) {
|
||||||
return fetch(`/kngil/bbs/adm_comp.php?action=base_code&main_cd=${mainCd}`)
|
return fetch(`/admin/api/company?action=base_code&main_cd=${mainCd}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
if (json.status !== 'success') {
|
if (json.status !== 'success') {
|
||||||
@@ -101,7 +101,7 @@ export function openProductPopup() {
|
|||||||
|
|
||||||
// 3. 브라우저 기본 확인창 사용 (가장 확실함)
|
// 3. 브라우저 기본 확인창 사용 (가장 확실함)
|
||||||
if (confirm(`선택한 ${ids.length}개의 상품을 삭제하시겠습니까?`)) {
|
if (confirm(`선택한 ${ids.length}개의 상품을 삭제하시겠습니까?`)) {
|
||||||
fetch('/kngil/bbs/adm_product_popup_delete.php', {
|
fetch('/admin/api/product/delete', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({ action: 'delete', ids: ids })
|
body: JSON.stringify({ action: 'delete', ids: ids })
|
||||||
@@ -171,7 +171,7 @@ export function openProductPopup() {
|
|||||||
console.log('INSERTS', inserts)
|
console.log('INSERTS', inserts)
|
||||||
console.log('UPDATES', updates)
|
console.log('UPDATES', updates)
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_product_popup_save.php', {
|
fetch('/admin/api/product/save', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -265,7 +265,7 @@ async function loadProductData() {
|
|||||||
try {
|
try {
|
||||||
w2ui.productGrid.lock('조회 중...', true);
|
w2ui.productGrid.lock('조회 중...', true);
|
||||||
|
|
||||||
const response = await fetch('/kngil/bbs/adm_product_popup.php'); // PHP 파일 호출
|
const response = await fetch('/admin/api/product'); // PHP 파일 호출
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
|
|
||||||
w2ui.productGrid.clear();
|
w2ui.productGrid.clear();
|
||||||
|
|||||||
@@ -117,7 +117,7 @@ async function loadPurchaseHistoryData(memberId) {
|
|||||||
searchParams.append('fbuy_dt', '');
|
searchParams.append('fbuy_dt', '');
|
||||||
searchParams.append('tbuy_dt', '');
|
searchParams.append('tbuy_dt', '');
|
||||||
|
|
||||||
const response = await fetch('/kngil/bbs/adm_purch_popup.php', {
|
const response = await fetch('/admin/api/purchase-history', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
|
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
|
||||||
body: searchParams
|
body: searchParams
|
||||||
|
|||||||
@@ -173,7 +173,7 @@ function addServiceFromProduct(p) {
|
|||||||
------------------------------------------------- */
|
------------------------------------------------- */
|
||||||
function loadExistingPurchase(memberId, buyDate) {
|
function loadExistingPurchase(memberId, buyDate) {
|
||||||
|
|
||||||
fetch(`/kngil/bbs/adm_service.php?member_id=${memberId}&buy_date=${buyDate}`)
|
fetch(`/admin/api/service?member_id=${memberId}&buy_date=${buyDate}`)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(json => {
|
.then(json => {
|
||||||
|
|
||||||
|
|||||||
@@ -288,7 +288,7 @@ function createProductList() {
|
|||||||
new w2grid({
|
new w2grid({
|
||||||
name: 'productList',
|
name: 'productList',
|
||||||
box: '#productList',
|
box: '#productList',
|
||||||
url: '/kngil/bbs/adm_product_popup.php',
|
url: '/admin/api/product',
|
||||||
columns: [
|
columns: [
|
||||||
{ field: 'itm_nm', text: '상품명', size: '120px' },
|
{ field: 'itm_nm', text: '상품명', size: '120px' },
|
||||||
{
|
{
|
||||||
@@ -413,7 +413,7 @@ function deleteServiceImmediately(row) {
|
|||||||
sq_no: row.sq_no
|
sq_no: row.sq_no
|
||||||
})
|
})
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_service.php', {
|
fetch('/admin/api/service', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -446,7 +446,7 @@ function isServiceItem(r) {
|
|||||||
------------------------------------------------- */
|
------------------------------------------------- */
|
||||||
function loadExistingPurchase(memberId, buyDate) {
|
function loadExistingPurchase(memberId, buyDate) {
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_service.php', {
|
fetch('/admin/api/service', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -606,7 +606,7 @@ function saveService(ctx) {
|
|||||||
_deleted: r._deleted || false
|
_deleted: r._deleted || false
|
||||||
}))
|
}))
|
||||||
|
|
||||||
fetch('/kngil/bbs/adm_service.php', {
|
fetch('/admin/api/service', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
|
|||||||
@@ -131,7 +131,7 @@ async function loadUseHistoryData(memberId = ''){
|
|||||||
searchParams.append('user_nm', sUnm);
|
searchParams.append('user_nm', sUnm);
|
||||||
searchParams.append('dept_nm', sDnm);
|
searchParams.append('dept_nm', sDnm);
|
||||||
|
|
||||||
const response = await fetch('/kngil/bbs/adm_use_history.php', {
|
const response = await fetch('/admin/api/use-history', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
|
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
|
||||||
body: searchParams
|
body: searchParams
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
@@ -894,6 +894,10 @@ function join() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function login() {
|
function login() {
|
||||||
|
if (typeof window.openOidcLogin === "function") {
|
||||||
|
window.openOidcLogin();
|
||||||
|
return;
|
||||||
|
}
|
||||||
$(".popup_wrap").hide();
|
$(".popup_wrap").hide();
|
||||||
//새로고침 없이 다시 팝업창 열었을때 자동 입력된 id, pw 제거
|
//새로고침 없이 다시 팝업창 열었을때 자동 입력된 id, pw 제거
|
||||||
$("#login_id").val("");
|
$("#login_id").val("");
|
||||||
@@ -906,6 +910,12 @@ function login() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function mypage01() {
|
function mypage01() {
|
||||||
|
const raw = (window.IDP_SERVICE_URL || "").trim();
|
||||||
|
if (raw) {
|
||||||
|
const url = raw.replace(/\/+$/, "") + "/profile";
|
||||||
|
window.open(url, "idp_profile", "noopener,noreferrer");
|
||||||
|
return;
|
||||||
|
}
|
||||||
$(".popup_wrap").hide();
|
$(".popup_wrap").hide();
|
||||||
$(".btn_close").show();
|
$(".btn_close").show();
|
||||||
$("#pop_mypage01").show(0, function () {
|
$("#pop_mypage01").show(0, function () {
|
||||||
|
|||||||
@@ -35,23 +35,6 @@ if (form) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// OIDC 로그인 처리
|
|
||||||
const btnOidc = document.getElementById('btn_oidc_login')
|
|
||||||
if (btnOidc) {
|
|
||||||
btnOidc.addEventListener('click', () => {
|
|
||||||
const width = 500;
|
|
||||||
const height = 600;
|
|
||||||
const left = (window.screen.width / 2) - (width / 2);
|
|
||||||
const top = (window.screen.height / 2) - (height / 2);
|
|
||||||
|
|
||||||
window.open(
|
|
||||||
'/kngil/auth/oidc-login.php',
|
|
||||||
'oidc_login_popup',
|
|
||||||
`width=${width},height=${height},left=${left},top=${top},scrollbars=yes`
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// 팝업으로부터의 메시지 수신 (로그인 성공 시 새로고침)
|
// 팝업으로부터의 메시지 수신 (로그인 성공 시 새로고침)
|
||||||
window.addEventListener('message', (event) => {
|
window.addEventListener('message', (event) => {
|
||||||
// 보안을 위해 실제 서비스에서는 event.origin을 체크하는 것이 좋습니다.
|
// 보안을 위해 실제 서비스에서는 event.origin을 체크하는 것이 좋습니다.
|
||||||
|
|||||||
@@ -20,7 +20,27 @@ function hide(id) {
|
|||||||
마이페이지 1단계 (비밀번호 인증)
|
마이페이지 1단계 (비밀번호 인증)
|
||||||
- 헤더에서 호출
|
- 헤더에서 호출
|
||||||
========================= */
|
========================= */
|
||||||
|
function getIdpProfileUrl() {
|
||||||
|
const raw = (window.IDP_SERVICE_URL || '').trim()
|
||||||
|
if (!raw) {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
return raw.replace(/\/+$/, '') + '/profile'
|
||||||
|
}
|
||||||
|
|
||||||
|
function openIdpProfile() {
|
||||||
|
const url = getIdpProfileUrl()
|
||||||
|
if (!url) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
window.open(url, 'idp_profile', 'noopener,noreferrer')
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
window.mypage01 = function () {
|
window.mypage01 = function () {
|
||||||
|
if (openIdpProfile()) {
|
||||||
|
return
|
||||||
|
}
|
||||||
if (!window.IS_LOGIN) {
|
if (!window.IS_LOGIN) {
|
||||||
if (typeof window.login === 'function') {
|
if (typeof window.login === 'function') {
|
||||||
window.login()
|
window.login()
|
||||||
|
|||||||
@@ -894,6 +894,10 @@ function join() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function login() {
|
function login() {
|
||||||
|
if (typeof window.openOidcLogin === "function") {
|
||||||
|
window.openOidcLogin();
|
||||||
|
return;
|
||||||
|
}
|
||||||
$(".popup_wrap").hide();
|
$(".popup_wrap").hide();
|
||||||
//새로고침 없이 다시 팝업창 열었을때 자동 입력된 id, pw 제거
|
//새로고침 없이 다시 팝업창 열었을때 자동 입력된 id, pw 제거
|
||||||
$("#login_id").val("");
|
$("#login_id").val("");
|
||||||
@@ -906,6 +910,12 @@ function login() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function mypage01() {
|
function mypage01() {
|
||||||
|
const raw = (window.IDP_SERVICE_URL || "").trim();
|
||||||
|
if (raw) {
|
||||||
|
const url = raw.replace(/\/+$/, "") + "/profile";
|
||||||
|
window.open(url, "idp_profile", "noopener,noreferrer");
|
||||||
|
return;
|
||||||
|
}
|
||||||
$(".popup_wrap").hide();
|
$(".popup_wrap").hide();
|
||||||
$(".btn_close").show();
|
$(".btn_close").show();
|
||||||
$("#pop_mypage01").show(0, function () {
|
$("#pop_mypage01").show(0, function () {
|
||||||
|
|||||||
0
kngil/log/join.log
Normal file → Executable file
0
kngil/log/join.log
Normal file → Executable file
@@ -3,8 +3,28 @@ if (session_status() === PHP_SESSION_NONE) {
|
|||||||
session_start();
|
session_start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
require_once __DIR__ . '/../bbs/env.php';
|
||||||
|
kngil_load_env_once(dirname(__DIR__, 2) . '/.env');
|
||||||
|
|
||||||
$isLogin = isset($_SESSION['login']);
|
$isLogin = isset($_SESSION['login']);
|
||||||
$auth = $_SESSION['login']['auth_bc'] ?? '';
|
$auth = $_SESSION['login']['auth_bc'] ?? '';
|
||||||
|
$idpServiceUrl = getenv('IDP_SERVICE_URL') ?: '';
|
||||||
|
$loginName = $_SESSION['login']['idp_name'] ?? ($_SESSION['login']['user_nm'] ?? '');
|
||||||
|
$loginEmail = $_SESSION['login']['idp_email'] ?? ($_SESSION['login']['email'] ?? '');
|
||||||
|
$displayName = $loginName ?: ($loginEmail ?: '');
|
||||||
|
$displayEmail = $loginEmail;
|
||||||
|
if ($displayName === 'Unknown') {
|
||||||
|
$displayName = '';
|
||||||
|
}
|
||||||
|
if ($displayEmail === 'Unknown') {
|
||||||
|
$displayEmail = '';
|
||||||
|
}
|
||||||
|
if ($displayName === $displayEmail) {
|
||||||
|
$displayEmail = '';
|
||||||
|
}
|
||||||
|
if ($displayName === '') {
|
||||||
|
$displayName = '사용자';
|
||||||
|
}
|
||||||
|
|
||||||
// 권한 그룹
|
// 권한 그룹
|
||||||
$isSuperAdmin = in_array($auth, ['BS100100', 'BS100200']);
|
$isSuperAdmin = in_array($auth, ['BS100100', 'BS100200']);
|
||||||
@@ -30,6 +50,53 @@ $isCompanyAdmin = in_array($auth, ['BS100100', 'BS100200', 'BS100300', 'BS100400
|
|||||||
.icon-btn:hover img {
|
.icon-btn:hover img {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
.auth-status {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-oidc-top {
|
||||||
|
--color-primary: #f95523;
|
||||||
|
--color-primary-border: #ca3f14;
|
||||||
|
background: var(--color-primary, #f95523);
|
||||||
|
border: 1px solid var(--color-primary-border, #ca3f14);
|
||||||
|
color: #fff;
|
||||||
|
padding: 8px 12px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
box-shadow: inset 0 0 0 1px rgba(255, 255, 255, 0.08);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-oidc-top:hover {
|
||||||
|
opacity: 0.85;
|
||||||
|
}
|
||||||
|
|
||||||
|
.auth-user {
|
||||||
|
color: #fff;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 600;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
.btn-logout-top {
|
||||||
|
--color-secondary: #3a3a3a;
|
||||||
|
--color-secondary-border: #2a2a2a;
|
||||||
|
background: var(--color-secondary, #3a3a3a);
|
||||||
|
border: 1px solid var(--color-secondary-border, #2a2a2a);
|
||||||
|
color: #fff;
|
||||||
|
padding: 8px 10px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-left: 8px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-logout-top:hover {
|
||||||
|
opacity: 0.85;
|
||||||
|
}
|
||||||
/* 툴팁 래퍼 */
|
/* 툴팁 래퍼 */
|
||||||
.tooltip {
|
.tooltip {
|
||||||
position: relative;
|
position: relative;
|
||||||
@@ -78,6 +145,7 @@ $isCompanyAdmin = in_array($auth, ['BS100100', 'BS100200', 'BS100300', 'BS100400
|
|||||||
</style>
|
</style>
|
||||||
<script>
|
<script>
|
||||||
window.IS_LOGIN = <?= isset($_SESSION['login']) ? 'true' : 'false' ?>;
|
window.IS_LOGIN = <?= isset($_SESSION['login']) ? 'true' : 'false' ?>;
|
||||||
|
window.IDP_SERVICE_URL = <?= json_encode($idpServiceUrl, JSON_UNESCAPED_SLASHES) ?>;
|
||||||
</script>
|
</script>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
@@ -85,16 +153,29 @@ $isCompanyAdmin = in_array($auth, ['BS100100', 'BS100200', 'BS100300', 'BS100400
|
|||||||
<div class="header-wrap">
|
<div class="header-wrap">
|
||||||
<header class="header">
|
<header class="header">
|
||||||
<h1>
|
<h1>
|
||||||
<a href="/kngil/skin/index.php">KNGIL</a>
|
<a href="/">KNGIL</a>
|
||||||
</h1>
|
</h1>
|
||||||
|
|
||||||
<div class="header-right">
|
<div class="header-right">
|
||||||
|
<div class="auth-status">
|
||||||
|
<?php if ($isLogin): ?>
|
||||||
|
<span class="auth-user">
|
||||||
|
<?= htmlspecialchars($displayName) ?>
|
||||||
|
<?php if ($displayEmail): ?>
|
||||||
|
(<?= htmlspecialchars($displayEmail) ?>)
|
||||||
|
<?php endif; ?>
|
||||||
|
</span>
|
||||||
|
<a href="/kngil/bbs/logout.php" class="btn-logout-top">로그아웃</a>
|
||||||
|
<?php else: ?>
|
||||||
|
<button type="button" class="btn-oidc-top" data-oidc-login>[바론 통합로그인]</button>
|
||||||
|
<?php endif; ?>
|
||||||
|
</div>
|
||||||
|
|
||||||
<?php if ($isLogin): ?>
|
<?php if ($isLogin): ?>
|
||||||
|
|
||||||
<!-- 통합 회원관리 (관리자/개발자만) -->
|
<!-- 통합 회원관리 (관리자/개발자만) -->
|
||||||
<?php if ($isSuperAdmin): ?>
|
<?php if ($isSuperAdmin): ?>
|
||||||
<a href="/kngil/skin/adm.php"
|
<a href="/admin"
|
||||||
class="icon-btn tooltip">
|
class="icon-btn tooltip">
|
||||||
<img src="/kngil/img/ico/ico_super_admin.svg" class="header-icon" alt="통합 회원관리">
|
<img src="/kngil/img/ico/ico_super_admin.svg" class="header-icon" alt="통합 회원관리">
|
||||||
<span class="tooltip-text">통합 회원관리</span>
|
<span class="tooltip-text">통합 회원관리</span>
|
||||||
@@ -103,7 +184,7 @@ $isCompanyAdmin = in_array($auth, ['BS100100', 'BS100200', 'BS100300', 'BS100400
|
|||||||
|
|
||||||
<!-- 회사 관리자 (관리자/개발자/메인/서브) -->
|
<!-- 회사 관리자 (관리자/개발자/메인/서브) -->
|
||||||
<?php if ($isCompanyAdmin): ?>
|
<?php if ($isCompanyAdmin): ?>
|
||||||
<a href="/kngil/skin/adm_comp.php"
|
<a href="/admin/company"
|
||||||
class="icon-btn tooltip">
|
class="icon-btn tooltip">
|
||||||
<img src="/kngil/img/ico/ico_company_admin.svg" class="header-icon" alt="회사 관리자">
|
<img src="/kngil/img/ico/ico_company_admin.svg" class="header-icon" alt="회사 관리자">
|
||||||
<span class="tooltip-text">회사 관리자</span>
|
<span class="tooltip-text">회사 관리자</span>
|
||||||
@@ -112,30 +193,23 @@ $isCompanyAdmin = in_array($auth, ['BS100100', 'BS100200', 'BS100300', 'BS100400
|
|||||||
|
|
||||||
<?php endif; ?>
|
<?php endif; ?>
|
||||||
|
|
||||||
<!-- 기존 사용자 메뉴 -->
|
<?php if ($isLogin): ?>
|
||||||
<div class="menu-box">
|
<!-- 기존 사용자 메뉴 -->
|
||||||
<button class="menu-user">
|
<div class="menu-box">
|
||||||
<img src="/kngil/img/ico/ico_user.svg" alt="user">
|
<button class="menu-user">
|
||||||
</button>
|
<img src="/kngil/img/ico/ico_user.svg" alt="user">
|
||||||
|
</button>
|
||||||
|
|
||||||
<ul class="menu-list">
|
<ul class="menu-list">
|
||||||
<!-- 비로그인 -->
|
<li>
|
||||||
<li class="<?= $isLogin ? 'd-none' : '' ?>">
|
<a class="my-join" href="javascript:mypage01()">마이페이지</a>
|
||||||
<a class="my-join" href="javascript:agreement();">회원가입</a>
|
</li>
|
||||||
</li>
|
<li>
|
||||||
<li class="<?= $isLogin ? 'd-none' : '' ?>">
|
<a class="my-login" href="/kngil/bbs/logout.php">로그아웃</a>
|
||||||
<a class="my-login" href="javascript:login();">로그인</a>
|
</li>
|
||||||
</li>
|
</ul>
|
||||||
|
</div>
|
||||||
<!-- 로그인 -->
|
<?php endif; ?>
|
||||||
<li class="<?= $isLogin ? '' : 'd-none' ?>">
|
|
||||||
<a class="my-join" href="javascript:mypage01()">마이페이지</a>
|
|
||||||
</li>
|
|
||||||
<li class="<?= $isLogin ? '' : 'd-none' ?>">
|
|
||||||
<a class="my-login" href="/kngil/bbs/logout.php">로그아웃</a>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<button class="menu-all">
|
<button class="menu-all">
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
<a href="/kngil/skin/index.php" class="btn-home-fixed" title="홈으로">
|
<a href="/" class="btn-home-fixed" title="홈으로">
|
||||||
<svg class="icon-home"
|
<svg class="icon-home"
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
viewBox="0 0 24 24"
|
viewBox="0 0 24 24"
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
<a href="/kngil/skin/index.php" class="btn-home-fixed" title="홈으로">
|
<a href="/" class="btn-home-fixed" title="홈으로">
|
||||||
<svg class="icon-home"
|
<svg class="icon-home"
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
viewBox="0 0 24 24"
|
viewBox="0 0 24 24"
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body class="page-admin">
|
<body class="page-admin">
|
||||||
<a href="/kngil/skin/index.php" class="btn-home-fixed" title="홈으로">
|
<a href="/" class="btn-home-fixed" title="홈으로">
|
||||||
<svg class="icon-home"
|
<svg class="icon-home"
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
viewBox="0 0 24 24"
|
viewBox="0 0 24 24"
|
||||||
|
|||||||
@@ -42,7 +42,6 @@
|
|||||||
<!-- //HEADER -->
|
<!-- //HEADER -->
|
||||||
|
|
||||||
<?php
|
<?php
|
||||||
include __DIR__ . "/pop_login.php";
|
|
||||||
include __DIR__ . "/pop_join.php";
|
include __DIR__ . "/pop_join.php";
|
||||||
include __DIR__ . "/pop_agreement.php";
|
include __DIR__ . "/pop_agreement.php";
|
||||||
include __DIR__ . "/pop_mypage01.php";
|
include __DIR__ . "/pop_mypage01.php";
|
||||||
@@ -153,7 +152,6 @@
|
|||||||
<script src="/kngil/js/index.js"></script>
|
<script src="/kngil/js/index.js"></script>
|
||||||
<script src="/kngil/js/mypage.js"></script>
|
<script src="/kngil/js/mypage.js"></script>
|
||||||
<script src="/kngil/js/join.js"></script>
|
<script src="/kngil/js/join.js"></script>
|
||||||
<script type="module" src="/kngil/js/login.js"></script>
|
|
||||||
<script src="/kngil/js/login_sms.js"></script>
|
<script src="/kngil/js/login_sms.js"></script>
|
||||||
<script>
|
<script>
|
||||||
AOS.init();
|
AOS.init();
|
||||||
|
|||||||
@@ -1,3 +1,8 @@
|
|||||||
|
<?php
|
||||||
|
if (session_status() === PHP_SESSION_NONE) {
|
||||||
|
session_start();
|
||||||
|
}
|
||||||
|
?>
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<!--index.php-->
|
<!--index.php-->
|
||||||
<html lang="ko">
|
<html lang="ko">
|
||||||
@@ -36,7 +41,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<?php
|
<?php
|
||||||
include __DIR__ . "/pop_login.php";
|
|
||||||
include __DIR__ . "/pop_join.php";
|
include __DIR__ . "/pop_join.php";
|
||||||
include __DIR__ . "/pop_agreement.php";
|
include __DIR__ . "/pop_agreement.php";
|
||||||
include __DIR__ . "/pop_mypage01.php";
|
include __DIR__ . "/pop_mypage01.php";
|
||||||
@@ -53,7 +57,6 @@
|
|||||||
<script src="/kngil/js/index.js"></script>
|
<script src="/kngil/js/index.js"></script>
|
||||||
<script src="/kngil/js/mypage.js"></script>
|
<script src="/kngil/js/mypage.js"></script>
|
||||||
<script src="/kngil/js/join.js"></script>
|
<script src="/kngil/js/join.js"></script>
|
||||||
<script type="module" src="/kngil/js/login.js"></script>
|
|
||||||
<script src="/kngil/js/login_sms.js"></script>
|
<script src="/kngil/js/login_sms.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
@@ -1,119 +0,0 @@
|
|||||||
<!-- popup_wrap -->
|
|
||||||
<div class="popup-wrap" id="pop_login" role="dialog" aria-labelledby="login_title" aria-modal="true">
|
|
||||||
<!-- popup_in -->
|
|
||||||
<div class="popup-in member">
|
|
||||||
<button class="btn-close" type="button" aria-label="팝업 닫기">
|
|
||||||
<img src="/kngil/img/ico/ico_close.svg" alt="닫기">
|
|
||||||
</button>
|
|
||||||
<!-- popup_container -->
|
|
||||||
<div class="popup-container login">
|
|
||||||
<!-- pop_header -->
|
|
||||||
<div class="pop-header">
|
|
||||||
<h2 class="tit" id="login_title">로그인</h2>
|
|
||||||
<p class="sub-txt">
|
|
||||||
KNGIL 로그인 시<br class="pc-only">
|
|
||||||
<em>다운로드 및 1:1문의 서비스를</em><br class="pc-only" /> 이용하실 수 있습니다.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<!-- //pop_header -->
|
|
||||||
<!-- pop_body -->
|
|
||||||
<div class="pop-body">
|
|
||||||
<div class="contents-wrap">
|
|
||||||
<ul class="tab-menu round">
|
|
||||||
<li class="tab-phone on"><span>휴대폰 인증</span></li>
|
|
||||||
<li class="tab-id"><span>아이디 로그인</span></li>
|
|
||||||
</ul>
|
|
||||||
<div class="tab-content phone">
|
|
||||||
<form action="" method="post" novalidate>
|
|
||||||
<div class="form-wrap">
|
|
||||||
<div class="input-box group">
|
|
||||||
<i class="phone" aria-hidden="true"></i>
|
|
||||||
<input type="tel" id="login_phone" name="userPhone" placeholder="휴대폰 번호 (예: 010-1234-5678)" required aria-label="휴대폰 번호">
|
|
||||||
<span class="timer d-none">03:00</span>
|
|
||||||
<!-- ✅ 재요청 버튼 -->
|
|
||||||
<button type="button"
|
|
||||||
class="btn-resend d-none"
|
|
||||||
id="sms_resend">
|
|
||||||
재요청
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div class="info-box d-none">
|
|
||||||
<p>
|
|
||||||
<em>인증 링크를 문자로 발송했습니다</em><br />
|
|
||||||
인증 문자가 오지 않으면 인증 링크를 재요청 해보세요
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div class="btn-wrap">
|
|
||||||
<button class="btn-full" id="sms_button" type="submit">인증 링크 요청</button>
|
|
||||||
</div>
|
|
||||||
<div class="btn-wrap">
|
|
||||||
<a class="go-signup" href="javascript:agreement();">
|
|
||||||
<span>회원가입</span><i class="arrow-r" aria-hidden="true"></i>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
<div class="tab-content id">
|
|
||||||
<form action="" method="post" novalidate>
|
|
||||||
<div class="form-wrap">
|
|
||||||
<div class="input-box group">
|
|
||||||
<i class="id" aria-hidden="true"></i>
|
|
||||||
<input type="text" id="login_id" name="userId" placeholder="아이디" required aria-label="아이디">
|
|
||||||
</div>
|
|
||||||
<div class="input-box group">
|
|
||||||
<i class="pw" aria-hidden="true"></i>
|
|
||||||
<input type="password" id="login_password" name="userPassword" placeholder="비밀번호" required aria-label="비밀번호">
|
|
||||||
</div>
|
|
||||||
<div class="btn-wrap">
|
|
||||||
<button class="btn-full" type="submit">로그인</button>
|
|
||||||
</div>
|
|
||||||
<div class="btn-wrap">
|
|
||||||
<a class="go-signup" href="javascript:agreement();">
|
|
||||||
<span>회원가입</span>
|
|
||||||
<i class="arrow-r" aria-hidden="true"></i>
|
|
||||||
</a>
|
|
||||||
<!-- 로그인 팝업 이동 -->
|
|
||||||
<a class="go-login" href="http://172.16.9.44/kngil/skin/popup/callback">
|
|
||||||
<span>로그인</span>
|
|
||||||
<i class="arrow-r" aria-hidden="true"></i>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<style>
|
|
||||||
.btn-oidc {
|
|
||||||
background-color: #000; /* IDP 브랜드 컬러에 맞춰 조정 가능 */
|
|
||||||
color: #fff;
|
|
||||||
border: none;
|
|
||||||
width: 100%;
|
|
||||||
padding: 12px;
|
|
||||||
border-radius: 4px;
|
|
||||||
margin-top: 10px;
|
|
||||||
cursor: pointer;
|
|
||||||
font-weight: bold;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
.btn-oidc:hover {
|
|
||||||
opacity: 0.8;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
<div class="btn-wrap">
|
|
||||||
<button type="button" class="btn-oidc" id="btn_oidc_login">
|
|
||||||
OIDC 통합 로그인
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
<!-- //pop_body -->
|
|
||||||
</div>
|
|
||||||
<!-- //popup_container -->
|
|
||||||
</div>
|
|
||||||
<!-- //popup_in -->
|
|
||||||
</div>
|
|
||||||
<!-- //popup_wrap -->
|
|
||||||
@@ -38,7 +38,6 @@
|
|||||||
|
|
||||||
|
|
||||||
<?php
|
<?php
|
||||||
include __DIR__ . "/pop_login.php";
|
|
||||||
include __DIR__ . "/pop_join.php";
|
include __DIR__ . "/pop_join.php";
|
||||||
include __DIR__ . "/pop_agreement.php";
|
include __DIR__ . "/pop_agreement.php";
|
||||||
include __DIR__ . "/pop_mypage01.php";
|
include __DIR__ . "/pop_mypage01.php";
|
||||||
@@ -347,7 +346,6 @@
|
|||||||
<script src="/kngil/js/index.js"></script>
|
<script src="/kngil/js/index.js"></script>
|
||||||
<script src="/kngil/js/mypage.js"></script>
|
<script src="/kngil/js/mypage.js"></script>
|
||||||
<script src="/kngil/js/join.js"></script>
|
<script src="/kngil/js/join.js"></script>
|
||||||
<script type="module" src="/kngil/js/login.js"></script>
|
|
||||||
<script src="/kngil/js/login_sms.js"></script>
|
<script src="/kngil/js/login_sms.js"></script>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
@@ -611,4 +609,3 @@
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -33,7 +33,6 @@
|
|||||||
<?php include __DIR__ . "/_header.php"; ?>
|
<?php include __DIR__ . "/_header.php"; ?>
|
||||||
<!-- //HEADER -->
|
<!-- //HEADER -->
|
||||||
<?php
|
<?php
|
||||||
include __DIR__ . "/pop_login.php";
|
|
||||||
include __DIR__ . "/pop_join.php";
|
include __DIR__ . "/pop_join.php";
|
||||||
include __DIR__ . "/pop_agreement.php";
|
include __DIR__ . "/pop_agreement.php";
|
||||||
include __DIR__ . "/pop_mypage01.php";
|
include __DIR__ . "/pop_mypage01.php";
|
||||||
@@ -259,7 +258,6 @@
|
|||||||
<script src="/kngil/js/index.js"></script>
|
<script src="/kngil/js/index.js"></script>
|
||||||
<script src="/kngil/js/mypage.js"></script>
|
<script src="/kngil/js/mypage.js"></script>
|
||||||
<script src="/kngil/js/join.js"></script>
|
<script src="/kngil/js/join.js"></script>
|
||||||
<script type="module" src="/kngil/js/login.js"></script>
|
|
||||||
<script src="/kngil/js/login_sms.js"></script>
|
<script src="/kngil/js/login_sms.js"></script>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
|||||||
@@ -35,7 +35,6 @@
|
|||||||
<?php include __DIR__ . "/_header.php"; ?>
|
<?php include __DIR__ . "/_header.php"; ?>
|
||||||
<!-- //HEADER -->
|
<!-- //HEADER -->
|
||||||
<?php
|
<?php
|
||||||
include __DIR__ . "/pop_login.php";
|
|
||||||
include __DIR__ . "/pop_join.php";
|
include __DIR__ . "/pop_join.php";
|
||||||
include __DIR__ . "/pop_agreement.php";
|
include __DIR__ . "/pop_agreement.php";
|
||||||
include __DIR__ . "/pop_mypage01.php";
|
include __DIR__ . "/pop_mypage01.php";
|
||||||
@@ -254,7 +253,6 @@
|
|||||||
<script src="/kngil/js/index.js"></script>
|
<script src="/kngil/js/index.js"></script>
|
||||||
<script src="/kngil/js/mypage.js"></script>
|
<script src="/kngil/js/mypage.js"></script>
|
||||||
<script src="/kngil/js/join.js"></script>
|
<script src="/kngil/js/join.js"></script>
|
||||||
<script type="module" src="/kngil/js/login.js"></script>
|
|
||||||
<script src="/kngil/js/login_sms.js"></script>
|
<script src="/kngil/js/login_sms.js"></script>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
|||||||
@@ -53,7 +53,6 @@
|
|||||||
?>
|
?>
|
||||||
|
|
||||||
<?php
|
<?php
|
||||||
include __DIR__ . "/pop_login.php";
|
|
||||||
include __DIR__ . "/pop_join.php";
|
include __DIR__ . "/pop_join.php";
|
||||||
include __DIR__ . "/pop_agreement.php";
|
include __DIR__ . "/pop_agreement.php";
|
||||||
include __DIR__ . "/pop_mypage01.php";
|
include __DIR__ . "/pop_mypage01.php";
|
||||||
@@ -154,7 +153,6 @@
|
|||||||
<script src="/kngil/js/index.js"></script>
|
<script src="/kngil/js/index.js"></script>
|
||||||
<script src="/kngil/js/mypage.js"></script>
|
<script src="/kngil/js/mypage.js"></script>
|
||||||
<script src="/kngil/js/join.js"></script>
|
<script src="/kngil/js/join.js"></script>
|
||||||
<script type="module" src="/kngil/js/login.js"></script>
|
|
||||||
<script src="/kngil/js/login_sms.js"></script>
|
<script src="/kngil/js/login_sms.js"></script>
|
||||||
|
|
||||||
<!-- CKEditor 표시용 초기화 (submit 관여 X) -->
|
<!-- CKEditor 표시용 초기화 (submit 관여 X) -->
|
||||||
|
|||||||
2366
kngil_plain.sql
Normal file
2366
kngil_plain.sql
Normal file
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user