Update auth

This commit is contained in:
Mulia Nasution 2023-05-29 15:48:01 +07:00
parent e577788c80
commit eb3599b351
7 changed files with 163 additions and 124 deletions

2
.env
View File

@ -1,3 +1,3 @@
VUE_APP_ROOT_URL=http://localhost:8080 VUE_APP_ROOT_URL=http://localhost:8080
VUE_APP_ROOT_API=http://localhost:9090/api VUE_APP_ROOT_API=http://localhost:9090/api
VUE_APP_PUBLIC_KEY=MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAx3MRA7zxvaWKrtmPl2hRJLFiyryvj0ZUlmWw9OZIgqwJUDBTsg5yFX4hCQrANV1yy5ibTqAn2APdNCdhGgp8R2YLWrUR2vVGbmnKXXzEDsFpT6cgo VUE_APP_PUBLIC_KEY="MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAkclHj0FpIPiB+XQL0P/mJVUdYbRXYS9vjSbVllzVteZgVDhK/srybDmAM2PSn8ZlakUa9bkBAGmiozaccDRJx5DJ/C80hIUcQneDKqWuY6rgnT3zKhjt3MHZK0I2VX5t7QgWHdh5Dni/D08JWwTkbnH+qhz8Wb4xlV0HokQB05YB00deuO09fr5Qw3GjgSSK8HA+euoajgijMbRPuReA4nkmBLhyTMryX/1h9vi8Wz9reGeiFdLZLECrcMmbLRF3d8iC/HISqBYdqdyjvGhXzGu19Gog1Qt8/qNh4fPmB0Le3EpfjRkcWrLqi22N7cGRAvagCJJAzqQVKe5S2fTVRwIDAQAB"

40
package-lock.json generated
View File

@ -9,6 +9,7 @@
"version": "0.1.0", "version": "0.1.0",
"hasInstallScript": true, "hasInstallScript": true,
"dependencies": { "dependencies": {
"axios": "^1.4.0",
"core-js": "^3.8.3", "core-js": "^3.8.3",
"devextreme": "^22.2.4", "devextreme": "^22.2.4",
"devextreme-vue": "^22.2.4", "devextreme-vue": "^22.2.4",
@ -3763,9 +3764,7 @@
"node_modules/asynckit": { "node_modules/asynckit": {
"version": "0.4.0", "version": "0.4.0",
"resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
"integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q=="
"dev": true,
"optional": true
}, },
"node_modules/at-least-node": { "node_modules/at-least-node": {
"version": "1.0.0", "version": "1.0.0",
@ -3837,6 +3836,29 @@
"dev": true, "dev": true,
"optional": true "optional": true
}, },
"node_modules/axios": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/axios/-/axios-1.4.0.tgz",
"integrity": "sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==",
"dependencies": {
"follow-redirects": "^1.15.0",
"form-data": "^4.0.0",
"proxy-from-env": "^1.1.0"
}
},
"node_modules/axios/node_modules/form-data": {
"version": "4.0.0",
"resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
"integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
"dependencies": {
"asynckit": "^0.4.0",
"combined-stream": "^1.0.8",
"mime-types": "^2.1.12"
},
"engines": {
"node": ">= 6"
}
},
"node_modules/babel-loader": { "node_modules/babel-loader": {
"version": "8.3.0", "version": "8.3.0",
"resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.3.0.tgz", "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.3.0.tgz",
@ -4768,8 +4790,6 @@
"version": "1.0.8", "version": "1.0.8",
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
"integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
"dev": true,
"optional": true,
"dependencies": { "dependencies": {
"delayed-stream": "~1.0.0" "delayed-stream": "~1.0.0"
}, },
@ -5609,8 +5629,6 @@
"version": "1.0.0", "version": "1.0.0",
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
"integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
"dev": true,
"optional": true,
"engines": { "engines": {
"node": ">=0.4.0" "node": ">=0.4.0"
} }
@ -7285,7 +7303,6 @@
"version": "1.15.2", "version": "1.15.2",
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz", "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
"integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==", "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==",
"dev": true,
"funding": [ "funding": [
{ {
"type": "individual", "type": "individual",
@ -9358,7 +9375,6 @@
"version": "1.52.0", "version": "1.52.0",
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
"dev": true,
"engines": { "engines": {
"node": ">= 0.6" "node": ">= 0.6"
} }
@ -9367,7 +9383,6 @@
"version": "2.1.35", "version": "2.1.35",
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
"dev": true,
"dependencies": { "dependencies": {
"mime-db": "1.52.0" "mime-db": "1.52.0"
}, },
@ -11068,6 +11083,11 @@
"node": ">= 0.10" "node": ">= 0.10"
} }
}, },
"node_modules/proxy-from-env": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
"integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
},
"node_modules/prr": { "node_modules/prr": {
"version": "1.0.1", "version": "1.0.1",
"resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz",

View File

@ -10,6 +10,7 @@
"postinstall": "npm run build-themes" "postinstall": "npm run build-themes"
}, },
"dependencies": { "dependencies": {
"axios": "^1.4.0",
"core-js": "^3.8.3", "core-js": "^3.8.3",
"devextreme": "^22.2.4", "devextreme": "^22.2.4",
"devextreme-vue": "^22.2.4", "devextreme-vue": "^22.2.4",

View File

@ -1,58 +1,54 @@
const defaultUser = { import * as Forge from 'node-forge'
email: 'titan@hadiyan.net', import axios from 'axios'
avatarUrl: 'https://js.devexpress.com/Demos/WidgetsGallery/JSDemos/images/employees/06.png' import { useRouter } from 'vue-router'
};
const router = useRouter()
export default { export default {
_user: defaultUser, _user: null,
loggedIn() { loggedIn() {
return !!this._user; let user = JSON.parse(window.localStorage.getItem('user'));
let access_token = window.localStorage.getItem('access_token');
if (user && access_token) {
axios.defaults.headers['authorization'] = `Bearer ${access_token}`;
return !!user;
}
return false;
}, },
async logIn(email, password) { async logIn(email, password) {
try { try {
// Send request
console.log(email, password);
// this._user = { ...defaultUser, email };
// return { const response = await axios.post('authentication/login', {
// isOk: true, email,
// data: this._user password: this.encrypt(password),
// }; });
const requestOptions = { const data = response.data;
method: "POST",
headers: { "Content-Type": "application/json" }, this._user = {
body: JSON.stringify({ email: email, password: password }), username: data.data.username,
email: data.data.email,
role: data.data.role,
avatarUrl: 'https://js.devexpress.com/Demos/WidgetsGallery/JSDemos/images/employees/06.png',
}; };
const response = await fetch('http://localhost:9090/api/authentication/login', requestOptions); window.localStorage.setItem('user', JSON.stringify(this._user))
const data = await response.json(); window.localStorage.setItem('access_token', response.data.data.accessToken)
window.localStorage.setItem('refresh_token', response.data.data.refreshToken)
console.log(data); axios.defaults.headers['authorization'] = `Bearer ${response.data.data.accessToken}`;
//console.log(data.data.length);
// this._user = {
// id: response.data.id,
// email: response.data.role,
// name: response.data.keterangan,
// role_id: response.data.id,
// };
if(data.status == 'sukses') {
const defaultUser = {
email: data.data.email,
avatarUrl: 'https://js.devexpress.com/Demos/WidgetsGallery/JSDemos/images/employees/06.png',
name: data.data.name,
};
this._user = { ...defaultUser };
if (response.status === 200) {
return { return {
isOk: true, isOk: true,
data: this._user data: this._user
}; };
} else { } else {
//--- jika gagal munculkan pesan gagal
return { return {
isOk: false, isOk: false,
message: "Authentication failed" message: "Authentication failed"
@ -70,16 +66,27 @@ export default {
async logOut() { async logOut() {
this._user = null; this._user = null;
window.localStorage.removeItem('user');
window.localStorage.removeItem('access_token');
window.localStorage.removeItem('refresh_token');
await axios.post('authentication/logout')
axios.defaults.headers['authorization'] = null;
router.push({ name: 'login-form' })
}, },
async getUser() { async getUser() {
try { try {
// Send request const user = JSON.parse(window.localStorage.getItem('user'));
if (user) {
return {
isOk: true,
data: user,
};
}
throw new Error(`Could not find current logged in user.`);
return {
isOk: true,
data: this._user
};
} }
catch { catch {
return { return {
@ -137,5 +144,23 @@ export default {
message: "Failed to create account" message: "Failed to create account"
}; };
} }
} },
encrypt(param) {
const publicKey = `-----BEGIN PUBLIC KEY-----
${process.env.VUE_APP_PUBLIC_KEY}
-----END PUBLIC KEY-----`
const rsa = Forge.pki.publicKeyFromPem(publicKey)
const encrypted = rsa.encrypt(param, 'RSA-OAEP', {
md: Forge.md.sha256.create(),
mgf1: {
md: Forge.md.sha1.create(),
},
});
return btoa(encrypted)
},
}; };

View File

@ -4,10 +4,13 @@ import './themes/generated/theme.additional.css';
import { createApp } from "vue"; import { createApp } from "vue";
import router from "./router"; import router from "./router";
import themes from "devextreme/ui/themes"; import themes from "devextreme/ui/themes";
import axios from 'axios'
import App from "./App"; import App from "./App";
import appInfo from "./app-info"; import appInfo from "./app-info";
axios.defaults.baseURL = `${process.env.VUE_APP_ROOT_API}`;
themes.initialized(() => { themes.initialized(() => {
const app = createApp(App); const app = createApp(App);
app.use(router); app.use(router);

View File

@ -121,7 +121,7 @@ const router = new createRouter({
{ {
path: "/:pathMatch(.*)*", path: "/:pathMatch(.*)*",
redirect: "/home" redirect: "/home"
}, },
{ {
path: "/user-page", path: "/user-page",
name: "user-page", name: "user-page",
@ -130,7 +130,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: User component: User
}, },
{ {
path: "/master-data/jenis-pengadaan", path: "/master-data/jenis-pengadaan",
name: "jenis-pengadaan", name: "jenis-pengadaan",
@ -139,7 +139,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: JenisPengadaan component: JenisPengadaan
}, },
{ {
path: "/master-data/unit-inisiator", path: "/master-data/unit-inisiator",
name: "unit-inisiator", name: "unit-inisiator",
@ -148,7 +148,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: UnitInisiator component: UnitInisiator
}, },
{ {
path: "/master-data/metode-pengadaan", path: "/master-data/metode-pengadaan",
name: "metode-pengadaan", name: "metode-pengadaan",
@ -157,7 +157,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: MetodePengadaan component: MetodePengadaan
}, },
{ {
path: "/master-data/metode-penyampaian", path: "/master-data/metode-penyampaian",
name: "metode-penyampaian", name: "metode-penyampaian",
@ -175,7 +175,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: JenisKontrak component: JenisKontrak
}, },
{ {
path: "/master-data/sumber-dana", path: "/master-data/sumber-dana",
name: "sumber-dana", name: "sumber-dana",
@ -184,7 +184,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: SumberDana component: SumberDana
}, },
{ {
path: "/master-data/strategi-pengadaan", path: "/master-data/strategi-pengadaan",
name: "strategi-pengadaan", name: "strategi-pengadaan",
@ -193,7 +193,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: StrategiPengadaan component: StrategiPengadaan
}, },
{ {
path: "/master-data/lokasi-pengadaan", path: "/master-data/lokasi-pengadaan",
name: "lokasi-pengadaan", name: "lokasi-pengadaan",
@ -202,7 +202,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: LokasiPengadaan component: LokasiPengadaan
}, },
{ {
path: "/master-data/supply-positioning-matrix", path: "/master-data/supply-positioning-matrix",
name: "supply-positioning-matrix", name: "supply-positioning-matrix",
@ -211,7 +211,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: SupplyPositioningMatrix component: SupplyPositioningMatrix
}, },
{ {
path: "/master-data/jenis-anggaran", path: "/master-data/jenis-anggaran",
name: "jenis-anggaran", name: "jenis-anggaran",
@ -229,7 +229,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: InstansiPage component: InstansiPage
}, },
{ {
path: "/master-data/bidang-page", path: "/master-data/bidang-page",
name: "bidang-page", name: "bidang-page",
@ -238,7 +238,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: BidangPage component: BidangPage
}, },
{ {
path: "/master-data/jabatan-page", path: "/master-data/jabatan-page",
name: "jabatan-page", name: "jabatan-page",
@ -256,7 +256,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: RolesConfig component: RolesConfig
}, },
{ {
path: "/master-config/users-config", path: "/master-config/users-config",
name: "users-config", name: "users-config",
@ -265,7 +265,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: UsersConfig component: UsersConfig
}, },
{ {
path: "/master-config/permissions-config", path: "/master-config/permissions-config",
name: "permissions-config", name: "permissions-config",
@ -274,7 +274,7 @@ const router = new createRouter({
layout: defaultLayout layout: defaultLayout
}, },
component: PermissionsConfig component: PermissionsConfig
}, },
{ {
path: "/rks-daftar", path: "/rks-daftar",
name: "rks-daftar", name: "rks-daftar",
@ -294,7 +294,7 @@ const router = new createRouter({
component: DrpPenyusunan component: DrpPenyusunan
}, },
], ],
history: createWebHashHistory() history: createWebHashHistory()
}); });

View File

@ -2,56 +2,42 @@
<div class="wrapper"> <div class="wrapper">
<div class="content-wrapper"> <div class="content-wrapper">
<form class="login-form" @submit.prevent="onSubmit"> <form class="login-form" @submit.prevent="onSubmit">
<dx-form :form-data="formData" :disabled="loading"> <dx-form :form-data="formData" :disabled="loading">
<dx-item <dx-item data-field="email" editor-type="dxTextBox"
data-field="email" :editor-options="{ stylingMode: 'filled', placeholder: 'Email', mode: 'email' }">
editor-type="dxTextBox" <dx-required-rule message="Dibutuhkan alamat email" />
:editor-options="{ stylingMode: 'filled', placeholder: 'Email', mode: 'email' }" <dx-email-rule message="Alamat email tidak valid" />
> <dx-label :visible="false" />
<dx-required-rule message="Dibutuhkan alamat email" /> </dx-item>
<dx-email-rule message="Alamat email tidak valid" /> <dx-item data-field='password' editor-type='dxTextBox'
<dx-label :visible="false" /> :editor-options="{ stylingMode: 'filled', placeholder: 'Password', mode: 'password' }">
</dx-item> <dx-required-rule message="Password harus diisi" />
<dx-item <dx-label :visible="false" />
data-field='password' </dx-item>
editor-type='dxTextBox' <dx-item data-field="rememberMe" editor-type="dxCheckBox"
:editor-options="{ stylingMode: 'filled', placeholder: 'Password', mode: 'password' }" :editor-options="{ text: 'Ingat saya', elementAttr: { class: 'form-text' } }">
> <dx-label :visible="false" />
<dx-required-rule message="Password harus diisi" /> </dx-item>
<dx-label :visible="false" /> <dx-button-item>
</dx-item> <dx-button-options width="100%" type="default" template="signInTemplate" :use-submit-behavior="true">
<dx-item </dx-button-options>
data-field="rememberMe" </dx-button-item>
editor-type="dxCheckBox" <dx-item>
:editor-options="{ text: 'Ingat saya', elementAttr: { class: 'form-text' } }" <template #default>
> <div class="link">
<dx-label :visible="false" /> <router-link to="/reset-password">Lupa password?</router-link>
</dx-item> </div>
<dx-button-item> </template>
<dx-button-options </dx-item>
width="100%" <template #signInTemplate>
type="default" <div>
template="signInTemplate" <span class="dx-button-text">
:use-submit-behavior="true" <dx-load-indicator v-if="loading" width="24px" height="24px" :visible="true" />
> <span v-if="!loading">Log In</span>
</dx-button-options> </span>
</dx-button-item> </div>
<dx-item> </template>
<template #default> </dx-form>
<div class="link">
<router-link to="/reset-password">Lupa password?</router-link>
</div>
</template>
</dx-item>
<template #signInTemplate>
<div>
<span class="dx-button-text">
<dx-load-indicator v-if="loading" width="24px" height="24px" :visible="true" />
<span v-if="!loading">Log In</span>
</span>
</div>
</template>
</dx-form>
</form> </form>
</div> </div>
</div> </div>
@ -80,8 +66,8 @@ export default {
const router = useRouter(); const router = useRouter();
const formData = reactive({ const formData = reactive({
email:"", email: "",
password:"" password: ""
}); });
const loading = ref(false); const loading = ref(false);
@ -93,11 +79,12 @@ export default {
const { email, password } = formData; const { email, password } = formData;
loading.value = true; loading.value = true;
const result = await auth.logIn(email, password); const result = await auth.logIn(email, password);
if (!result.isOk) { if (!result.isOk) {
loading.value = false; loading.value = false;
notify(result.message, "error", 2000); notify(result.message, "error", 2000);
} else { } else {
router.push(route.query.redirect || "/home"); router.push(route.query?.redirect || "/home");
} }
} }
@ -141,5 +128,8 @@ export default {
} }
} }
html, body { height: 100%; } html,
body {
height: 100%;
}
</style> </style>