Skip to content

Commit

Permalink
Merge branch 'feature/performance-enhancement-jan-30-2020' into develop
Browse files Browse the repository at this point in the history
  • Loading branch information
johnnymillergh committed Feb 11, 2020
2 parents d1edb9f + 6104d5b commit 4527a1d
Show file tree
Hide file tree
Showing 19 changed files with 532 additions and 97 deletions.
2 changes: 1 addition & 1 deletion .env
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,4 @@
VUE_APP_PACKAGE_JSON=''

# TAG must be corresponding with the version tag in package.json, need to modify it when new version releases
TAG=1.0.2
TAG=1.1.0-beta
4 changes: 2 additions & 2 deletions .env.production
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,6 @@ VUE_APP_RUN_ON_DOCKER = 'true'
# Left blank if the environment is not development.
VUE_APP_BASE_URL = ''
# Base api
VUE_APP_BASE_API = 'http://106.13.32.85:8081/jm-spring-boot-template-prod'
VUE_APP_BASE_API = 'http://172.11.239.12:8080/exrx-net-crawler-server-prod'
# Resource base api for picture, video ect.
VUE_APP_RESOURCE_BASE_API = 'http://106.13.32.85:8081/jm-spring-boot-template-prod'
VUE_APP_RESOURCE_BASE_API = 'http://172.11.239.12:8080/exrx-net-crawler-server-prod'
62 changes: 30 additions & 32 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "exrx-net-crawler",
"version": "1.0.2",
"version": "1.1.0-beta",
"license": "Apache-2.0",
"description": "The Crawler for ExRx.net",
"author": {
Expand Down Expand Up @@ -35,55 +35,53 @@
"go:travis": "npm run lint && npm run test:unit && npm run build:prod"
},
"dependencies": {
"@mdi/font": "4.7.95",
"@mdi/js": "4.7.95",
"@types/cheerio": "0.22.15",
"@mdi/font": "4.9.95",
"@mdi/js": "4.9.95",
"@types/cheerio": "0.22.16",
"@types/lodash": "4.14.149",
"axios": "0.19.0",
"axios": "0.19.2",
"axios-mock-adapter": "1.17.0",
"cheerio": "1.0.0-rc.3",
"class-validator": "0.11.0",
"core-js": "3.6.1",
"core-js": "3.6.4",
"lodash": "4.17.15",
"mysql": "2.17.1",
"reflect-metadata": "0.1.13",
"roboto-fontface": "*",
"vue": "2.6.11",
"vue-class-component": "7.1.0",
"vue-property-decorator": "8.3.0",
"vue-router": "3.1.3",
"vuetify": "2.2.1",
"vue-class-component": "7.2.2",
"vue-property-decorator": "8.4.0",
"vue-router": "3.1.5",
"vuetify": "2.2.11",
"vuex": "3.1.2"
},
"devDependencies": {
"@babel/plugin-proposal-optional-chaining": "7.7.5",
"@types/jest": "^24.0.19",
"@typescript-eslint/eslint-plugin": "2.14.0",
"@typescript-eslint/parser": "2.14.0",
"@vue/cli-plugin-babel": "4.1.2",
"@vue/cli-plugin-eslint": "4.1.2",
"@vue/cli-plugin-router": "4.1.2",
"@vue/cli-plugin-typescript": "4.1.2",
"@vue/cli-plugin-unit-jest": "4.1.2",
"@vue/cli-plugin-vuex": "4.1.2",
"@vue/cli-service": "4.1.2",
"@vue/eslint-config-standard": "5.0.1",
"@babel/plugin-proposal-optional-chaining": "7.8.3",
"@types/jest": "24.9.1",
"@typescript-eslint/eslint-plugin": "2.19.0",
"@typescript-eslint/parser": "2.19.0",
"@vue/cli-plugin-babel": "4.2.2",
"@vue/cli-plugin-eslint": "4.2.2",
"@vue/cli-plugin-router": "4.2.2",
"@vue/cli-plugin-typescript": "4.2.2",
"@vue/cli-plugin-unit-jest": "4.2.2",
"@vue/cli-plugin-vuex": "4.2.2",
"@vue/cli-service": "4.2.2",
"@vue/eslint-config-standard": "5.1.1",
"@vue/eslint-config-typescript": "5.0.1",
"@vue/test-utils": "1.0.0-beta.29",
"@vue/test-utils": "1.0.0-beta.31",
"babel-eslint": "10.0.3",
"eslint": "6.8.0",
"eslint-plugin-import": "2.19.1",
"eslint-plugin-import": "2.20.1",
"eslint-plugin-node": "11.0.0",
"eslint-plugin-promise": "4.2.1",
"eslint-plugin-standard": "4.0.1",
"eslint-plugin-vue": "6.1.2",
"eslint-plugin-vue-libs": "4.0.0",
"lint-staged": "9.5.0",
"node-sass": "4.13.0",
"sass": "1.24.0",
"sass-loader": "8.0.0",
"typescript": "3.7.4",
"vue-cli-plugin-vuetify": "2.0.3",
"lint-staged": "10.0.7",
"node-sass": "4.13.1",
"sass": "1.25.0",
"sass-loader": "8.0.2",
"typescript": "3.7.5",
"vue-cli-plugin-vuetify": "2.0.4",
"vue-template-compiler": "2.6.11",
"vuetify-loader": "1.4.3"
},
Expand Down
7 changes: 7 additions & 0 deletions proxy.config.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,11 @@
module.exports = {
'/common': {
target: `${process.env.VUE_APP_BASE_URL}`,
changeOrigin: true,
pathRewrite: {
'^/common': `/${process.env.VUE_APP_BASE_API}/common`
}
},
'/test-table': {
target: `${process.env.VUE_APP_BASE_URL}`,
changeOrigin: true,
Expand Down
25 changes: 19 additions & 6 deletions src/components/HelloWorld.vue
Original file line number Diff line number Diff line change
Expand Up @@ -18,40 +18,53 @@
<v-divider style="margin-bottom: 24px"/>
<v-row>
<v-btn v-debounced-click="handelClickTest" block color="primary">Test Back-end Service</v-btn>
<v-btn @click="handleClickOpenNewPage" block>Open New Page</v-btn>
</v-row>
<v-row>{{ response }}</v-row>
<v-row>Response: {{ response }}</v-row>
<v-row>Value from sub window: {{ JSON.stringify(valueFromSubWindow) }}</v-row>
</v-container>
</template>

<!--suppress JSUnusedLocalSymbols, JSUnusedGlobalSymbols -->
<script lang="ts">
import { Component, Prop, Vue } from 'vue-property-decorator'
import { AppUtil } from '@/utils/app-util'
import { GetByIdPayload } from '@/domain/test-table/get-by-id-payload'
import { testTableApi } from '@/requests/test-table-api'
import { commonApi } from '@/requests/common-api'

@Component
export default class HelloWorld extends Vue {
@Prop() private msg!: string
private version = ''
private environment = ''
private response = null as any
private valueFromSubWindow = null as any

async mounted () {
this.version = AppUtil.getVersionInfo()
this.environment = `${process.env.NODE_ENV} (${process.env.VUE_APP_ENV})`
}

async handelClickTest (): Promise<void> {
const getByIdPayload = new GetByIdPayload()
getByIdPayload.id = 1
try {
this.response = await testTableApi.getById(getByIdPayload)
this.response = await commonApi.appInfo()
this.$toast.success('Succeed to interact with back-end server `exrx-net-crawler-server`')
} catch (error) {
console.error('Error occurred when sending request `getById`!', error)
this.$toast.error(error.message)
}
}

handleClickOpenNewPage () {
this.openWindow(this, '/second-page', {
callback: 'logCallback',
windowTarget: 'secondPage',
version: this.version
})
}

logCallback (value: any) {
console.info('logCallback', value)
this.valueFromSubWindow = value
}
}
</script>
2 changes: 2 additions & 0 deletions src/main.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,10 @@ import '@/directives/throttled-click'
import '@/directives/debounced-click'
import { listenToColorScheme } from '@/plugins/adaptive-color-scheme'
// import '@/plugins/axios-mock-adapter'
import { MultiWindow } from '@/mixins/multi-window'

Vue.config.productionTip = false
Vue.mixin(MultiWindow)

const vueInstance = new Vue({
router,
Expand Down
80 changes: 80 additions & 0 deletions src/mixins/multi-window/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,80 @@
import { Component, Vue } from 'vue-property-decorator'
// eslint-disable-next-line no-unused-vars
import { MultiWindowOptions } from '@/mixins/multi-window/multi-window-options'

/**
* Multi window mixin
*
* Can i use mixins global in vue? i am use typescript with vue
* @see <a href='https://github.com/kaorun343/vue-property-decorator/issues/226#issuecomment-515568960'>GitHub</a>
*/
@Component
export class MultiWindow extends Vue {
private $multiWindowOptions: MultiWindowOptions = {
delayClosingWindow: true,
callback: ''
}

mounted () {
this.$data.$multiWindowOptions.delayClosingWindow = this.$route?.query?.delayClosingWindow === 'true'
this.$data.$multiWindowOptions.callback = this.$route?.query?.callback
}

openWindow (context: Vue, url: string, multiWindowOptions?: MultiWindowOptions): void {
// Set default delayClosingWindow as true
if (multiWindowOptions?.delayClosingWindow === undefined) {
// @ts-ignore
multiWindowOptions.delayClosingWindow = true
}
const path = /\/$/.test(url) ? url : `${url}/`
// the value of process.env.BASE_URL is equal to the `publicPath` configured in vue.config.js
const baseUrl = process.env.BASE_URL
const target = /^(http|https):\/\//.test(path) ? url : `${baseUrl}#${url}`
// Passing information to opened window by query string
let queryString = ''
for (const key in multiWindowOptions) {
// Determines whether an object has a property with the specified name.
// eslint-disable-next-line no-prototype-builtins
if (!multiWindowOptions.hasOwnProperty(key)) {
break
}
const val = ((multiWindowOptions[key] === null) || (multiWindowOptions[key] === undefined) ? '' : multiWindowOptions[key])
queryString += queryString === '' ? `?${key}=${val}` : `&${key}=${val}`
}
setTimeout(() => {
const newWindow = window.open(`${target}${queryString}`, multiWindowOptions?.windowTarget)
if (!newWindow) {
window.alert('Please give us permission to open a new page!')
throw new Error('Failed to gain permission to open a new page!')
} else {
newWindow.opener.$vue = context
}
}, 600)
}

windowBack (argument?: any): void {
const context = window?.opener?.$vue
const callback = this.$data.$multiWindowOptions.callback
const delayClosingWindow = this.$data.$multiWindowOptions.delayClosingWindow
if (!context) {
this.$toast.error('ERROR: Cannot find context!')
// window.alert('ERROR: Cannot find context!')
throw new Error('Cannot find context!')
}
// Check whether opener's callback is valid
if (callback && typeof context[callback] === 'function') {
context[callback](argument)
} else {
this.$toast.error('ERROR: Cannot find callback!')
// window.alert('ERROR: Cannot find callback!')
throw new Error('Cannot find callback!')
}
if (delayClosingWindow) {
setTimeout(() => {
window.close()
}, 300)
} else {
window.close()
}
}
}
25 changes: 25 additions & 0 deletions src/mixins/multi-window/multi-window-options.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
/**
* Multi Window Options
*/
export interface MultiWindowOptions {
/**
* A DOMString specifying the name of the browsing context (window, <iframe> or tab) into which to load the specified resource;
* if the name doesn't indicate an existing context, a new window is created and is given the name specified by windowName.
*/
windowTarget?: string

/**
* Whether delay closing opened window for 300 ms. Default: true
*/
delayClosingWindow?: boolean

/**
* Vue instance of opener callback function name
*/
callback?: string

/**
* Other data
*/
[key: string]: any
}
17 changes: 9 additions & 8 deletions src/plugins/axios/axios-for-inter-communication.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,10 @@
* @date 1/2/20 9:15 AM
*/
// eslint-disable-next-line no-unused-vars
import Axios, { AxiosRequestConfig, AxiosResponse, Canceler, ResponseType } from 'axios'
import * as Cancellation from '@/plugins/axios/cancellation'
import Axios, { AxiosRequestConfig, AxiosResponse, ResponseType } from 'axios'
// import * as Cancellation from '@/plugins/axios/cancellation'
import { HttpStatus } from '@/constants/http-status'
// import { AxiosUtil } from '@/utils/axios-util'

/**
CORS Anywhere is a NodeJS reverse proxy which adds CORS headers to the proxied request.
Expand Down Expand Up @@ -65,13 +66,13 @@ export const service = Axios.create({
service.interceptors.request.use(
async (axiosRequestConfig: AxiosRequestConfig) => {
// Cancel and remove same request before sending upcoming request.
Cancellation.cancelAndRemoveSamePendingRequest(axiosRequestConfig)
// Cancellation.cancelAndRemoveSamePendingRequest(axiosRequestConfig)
// Configure cancelToken for request
axiosRequestConfig.cancelToken = new Cancellation.CancelToken((cancel: Canceler) => {
const requestToken = `${axiosRequestConfig?.url?.split('?')[0]}::${axiosRequestConfig.method}::${JSON.stringify(axiosRequestConfig.params)}`
const pendingRequest = new Cancellation.PendingRequest(requestToken, cancel)
Cancellation.pendingRequestList.push(pendingRequest)
})
// axiosRequestConfig.cancelToken = new Cancellation.CancelToken((cancel: Canceler) => {
// const requestToken = AxiosUtil.getRequestToken(axiosRequestConfig)
// const pendingRequest = new Cancellation.PendingRequest(requestToken, cancel)
// Cancellation.pendingRequestList.push(pendingRequest)
// })
return axiosRequestConfig
},
(error: any) => {
Expand Down
31 changes: 20 additions & 11 deletions src/plugins/axios/axios-for-intra-communication.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,13 @@
* @date 1/2/20 9:15 AM
*/
// eslint-disable-next-line no-unused-vars
import Axios, { AxiosRequestConfig, AxiosResponse, Canceler, ResponseType } from 'axios'
import Axios, { AxiosRequestConfig, AxiosResponse, ResponseType } from 'axios'
import { validate } from 'class-validator'
import { ClassValidationUtil } from '@/utils/class-validation-util'
import * as Cancellation from '@/plugins/axios/cancellation'
// import * as Cancellation from '@/plugins/axios/cancellation'
import { HttpStatus } from '@/constants/http-status'
import { ResponseBody } from '@/plugins/axios/response-body'
// import { AxiosUtil } from '@/utils/axios-util'

// 1. Create an axios instance.
export const service = Axios.create({
Expand All @@ -35,19 +36,27 @@ service.interceptors.request.use(
if (axiosRequestConfig?.params) {
const validation = await validate(axiosRequestConfig.params)
if (validation.length > 0) {
console.error('Validation failed! Validation:', validation)
console.error('Validation failed! Error message:', ClassValidationUtil.getAllValidationError(validation))
throw new Error(`Validation failed! The 1st error: ${ClassValidationUtil.getFirstValidationError(validation)}`)
console.error('Params validation failed! Validation:', validation)
console.error('Params validation failed! Error message:', ClassValidationUtil.getAllValidationError(validation))
throw new Error(`Params validation failed! The 1st error: ${ClassValidationUtil.getFirstValidationError(validation)}`)
}
}
if (axiosRequestConfig?.data) {
const validation = await validate(axiosRequestConfig.data)
if (validation.length > 0) {
console.error('Data validation failed! Validation:', validation)
console.error('Data validation failed! Error message:', ClassValidationUtil.getAllValidationError(validation))
throw new Error(`Data validation failed! The 1st error: ${ClassValidationUtil.getFirstValidationError(validation)}`)
}
}
// Cancel and remove same request before sending upcoming request.
Cancellation.cancelAndRemoveSamePendingRequest(axiosRequestConfig)
// Cancellation.cancelAndRemoveSamePendingRequest(axiosRequestConfig)
// Configure cancelToken for request
axiosRequestConfig.cancelToken = new Cancellation.CancelToken((cancel: Canceler) => {
const requestToken = `${axiosRequestConfig?.url?.split('?')[0]}::${axiosRequestConfig.method}::${JSON.stringify(axiosRequestConfig.params)}`
const pendingRequest = new Cancellation.PendingRequest(requestToken, cancel)
Cancellation.pendingRequestList.push(pendingRequest)
})
// axiosRequestConfig.cancelToken = new Cancellation.CancelToken((cancel: Canceler) => {
// const requestToken = AxiosUtil.getRequestToken(axiosRequestConfig)
// const pendingRequest = new Cancellation.PendingRequest(requestToken, cancel)
// Cancellation.pendingRequestList.push(pendingRequest)
// })
return axiosRequestConfig
},
(error: any) => {
Expand Down
3 changes: 2 additions & 1 deletion src/plugins/axios/cancellation.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// eslint-disable-next-line no-unused-vars
import Axios, { AxiosRequestConfig, Canceler } from 'axios'
import { AxiosUtil } from '@/utils/axios-util'

export const CancelToken = Axios.CancelToken
export const pendingRequestList: PendingRequest[] = []
Expand Down Expand Up @@ -32,7 +33,7 @@ export class PendingRequest {

export const cancelAndRemoveSamePendingRequest = (axiosRequestConfig: AxiosRequestConfig): void => {
pendingRequestList.forEach((pendingRequest, index) => {
const requestToken = `${axiosRequestConfig?.url?.split('?')[0]}::${axiosRequestConfig.method}::${JSON.stringify(axiosRequestConfig.params)}`
const requestToken = AxiosUtil.getRequestToken(axiosRequestConfig)
if (pendingRequest.requestToken === requestToken) {
// Execute cancellation of this pending request.
pendingRequest.cancelExecutor(`Cancelled Axios request. Request token: ${requestToken}`)
Expand Down
Loading

0 comments on commit 4527a1d

Please sign in to comment.