| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  | if (!process.env.CI) require('dotenv-safe').load(); | 
					
						
							| 
									
										
										
										
											2018-07-11 11:02:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-22 16:21:42 +02:00
										 |  |  | const assert = require('node:assert'); | 
					
						
							| 
									
										
										
										
											2021-08-12 10:34:49 -07:00
										 |  |  | const got = require('got'); | 
					
						
							| 
									
										
										
										
											2019-10-31 14:25:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  | const { Octokit } = require('@octokit/rest'); | 
					
						
							|  |  |  | const octokit = new Octokit({ | 
					
						
							|  |  |  |   auth: process.env.ELECTRON_GITHUB_TOKEN | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  | const BUILD_APPVEYOR_URL = 'https://ci.appveyor.com/api/builds'; | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  | const GH_ACTIONS_PIPELINE_URL = 'https://github.com/electron/electron/actions'; | 
					
						
							|  |  |  | const GH_ACTIONS_API_URL = '/repos/electron/electron/actions'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const GH_ACTIONS_WAIT_TIME = process.env.GH_ACTIONS_WAIT_TIME || 30000; | 
					
						
							| 
									
										
										
										
											2017-11-02 15:26:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-09 15:29:44 -04:00
										 |  |  | const appVeyorJobs = { | 
					
						
							| 
									
										
										
										
											2019-05-23 16:54:34 -04:00
										 |  |  |   'electron-x64': 'electron-x64-release', | 
					
						
							| 
									
										
										
										
											2019-09-04 14:24:46 -04:00
										 |  |  |   'electron-ia32': 'electron-ia32-release', | 
					
						
							|  |  |  |   'electron-woa': 'electron-woa-release' | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-07-09 15:29:44 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  | const ghActionsPublishWorkflows = [ | 
					
						
							| 
									
										
										
										
											2024-07-01 03:33:12 -07:00
										 |  |  |   'linux-publish', | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  |   'macos-publish' | 
					
						
							|  |  |  | ]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  | let jobRequestedCount = 0; | 
					
						
							| 
									
										
										
										
											2019-09-04 14:24:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-28 20:26:00 -07:00
										 |  |  | async function makeRequest ({ auth, username, password, url, headers, body, method }) { | 
					
						
							| 
									
										
										
										
											2021-08-13 10:37:30 -07:00
										 |  |  |   const clonedHeaders = { | 
					
						
							|  |  |  |     ...(headers || {}) | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2022-06-28 20:26:00 -07:00
										 |  |  |   if (auth?.bearer) { | 
					
						
							| 
									
										
										
										
											2021-08-13 10:37:30 -07:00
										 |  |  |     clonedHeaders.Authorization = `Bearer ${auth.bearer}`; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-06-28 20:26:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const options = { | 
					
						
							| 
									
										
										
										
											2021-08-13 10:37:30 -07:00
										 |  |  |     headers: clonedHeaders, | 
					
						
							| 
									
										
										
										
											2021-08-12 10:34:49 -07:00
										 |  |  |     body, | 
					
						
							| 
									
										
										
										
											2022-06-28 20:26:00 -07:00
										 |  |  |     method | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (username || password) { | 
					
						
							|  |  |  |     options.username = username; | 
					
						
							|  |  |  |     options.password = password; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const response = await got(url, options); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 10:25:17 -07:00
										 |  |  |   if (response.statusCode < 200 || response.statusCode >= 300) { | 
					
						
							| 
									
										
										
										
											2021-08-12 10:34:49 -07:00
										 |  |  |     console.error('Error: ', `(status ${response.statusCode})`, response.body); | 
					
						
							|  |  |  |     throw new Error(`Unexpected status code ${response.statusCode} from ${url}`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return JSON.parse(response.body); | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  | async function githubActionsCall (targetBranch, workflowName, options) { | 
					
						
							|  |  |  |   console.log(`Triggering GitHub Actions to run build job: ${workflowName} on branch: ${targetBranch} with release flag.`); | 
					
						
							|  |  |  |   const buildRequest = { | 
					
						
							|  |  |  |     branch: targetBranch, | 
					
						
							|  |  |  |     parameters: {} | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   if (options.ghRelease) { | 
					
						
							|  |  |  |     buildRequest.parameters['upload-to-storage'] = '0'; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     buildRequest.parameters['upload-to-storage'] = '1'; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   buildRequest.parameters[`run-${workflowName}`] = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   jobRequestedCount++; | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     const commits = await octokit.repos.listCommits({ | 
					
						
							|  |  |  |       owner: 'electron', | 
					
						
							|  |  |  |       repo: 'electron', | 
					
						
							|  |  |  |       sha: targetBranch, | 
					
						
							|  |  |  |       per_page: 5 | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     if (!commits.data.length) { | 
					
						
							|  |  |  |       console.error('Could not fetch most recent commits for GitHub Actions, returning early'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await octokit.request(`POST ${GH_ACTIONS_API_URL}/workflows/${workflowName}.yml/dispatches`, { | 
					
						
							| 
									
										
										
										
											2024-07-01 03:33:12 -07:00
										 |  |  |       ref: `refs/tags/${options.newVersion}`, | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  |       inputs: { | 
					
						
							|  |  |  |         ...buildRequest.parameters | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       headers: { | 
					
						
							|  |  |  |         'X-GitHub-Api-Version': '2022-11-28' | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const runNumber = await getGitHubActionsRun(workflowName, commits.data[0].sha); | 
					
						
							|  |  |  |     if (runNumber === -1) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     console.log(`GitHub Actions release build pipeline ${runNumber} for ${workflowName} triggered.`); | 
					
						
							|  |  |  |     const runUrl = `${GH_ACTIONS_PIPELINE_URL}/runs/${runNumber}`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (options.runningPublishWorkflows) { | 
					
						
							|  |  |  |       console.log(`GitHub Actions release workflow request for ${workflowName} successful.  Check ${runUrl} for status.`); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       console.log(`GitHub Actions release build workflow running at ${GH_ACTIONS_PIPELINE_URL}/runs/${runNumber} for ${workflowName}.`); | 
					
						
							|  |  |  |       console.log(`GitHub Actions release build request for ${workflowName} successful.  Check ${runUrl} for status.`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } catch (err) { | 
					
						
							|  |  |  |     console.log('Error calling GitHub Actions: ', err); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function getGitHubActionsRun (workflowId, headCommit) { | 
					
						
							|  |  |  |   let runNumber = 0; | 
					
						
							|  |  |  |   let actionRun; | 
					
						
							|  |  |  |   while (runNumber === 0) { | 
					
						
							|  |  |  |     const actionsRuns = await octokit.request(`GET ${GH_ACTIONS_API_URL}/workflows/${workflowId}.yml/runs`, { | 
					
						
							|  |  |  |       headers: { | 
					
						
							|  |  |  |         'X-GitHub-Api-Version': '2022-11-28' | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     if (!actionsRuns.data.workflow_runs.length) { | 
					
						
							|  |  |  |       console.log(`No current workflow_runs found for ${workflowId}, response was: ${actionsRuns.data.workflow_runs}`); | 
					
						
							|  |  |  |       runNumber = -1; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const run of actionsRuns.data.workflow_runs) { | 
					
						
							|  |  |  |       if (run.head_sha === headCommit) { | 
					
						
							|  |  |  |         console.log(`GitHub Actions run ${run.html_url} found for ${headCommit}, waiting on status.`); | 
					
						
							|  |  |  |         actionRun = run; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (actionRun) { | 
					
						
							|  |  |  |       switch (actionRun.status) { | 
					
						
							|  |  |  |         case 'in_progress': | 
					
						
							|  |  |  |         case 'pending': | 
					
						
							|  |  |  |         case 'queued': | 
					
						
							|  |  |  |         case 'requested': | 
					
						
							|  |  |  |         case 'waiting': { | 
					
						
							|  |  |  |           if (actionRun.id && !isNaN(actionRun.id)) { | 
					
						
							|  |  |  |             console.log(`GitHub Actions run ${actionRun.status} for ${actionRun.html_url}.`); | 
					
						
							|  |  |  |             runNumber = actionRun.id; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case 'action_required': | 
					
						
							|  |  |  |         case 'cancelled': | 
					
						
							|  |  |  |         case 'failure': | 
					
						
							|  |  |  |         case 'skipped': | 
					
						
							|  |  |  |         case 'timed_out': | 
					
						
							|  |  |  |         case 'failed': { | 
					
						
							|  |  |  |           console.log(`Error workflow run returned a status of ${actionRun.status} for ${actionRun.html_url}`); | 
					
						
							|  |  |  |           runNumber = -1; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       await new Promise(resolve => setTimeout(resolve, GH_ACTIONS_WAIT_TIME)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return runNumber; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-09 15:29:44 -04:00
										 |  |  | async function callAppVeyor (targetBranch, job, options) { | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |   console.log(`Triggering AppVeyor to run build job: ${job} on branch: ${targetBranch} with release flag.`); | 
					
						
							| 
									
										
										
										
											2018-09-27 15:38:06 +10:00
										 |  |  |   const environmentVariables = { | 
					
						
							| 
									
										
										
										
											2020-03-30 15:46:42 -04:00
										 |  |  |     ELECTRON_RELEASE: 1, | 
					
						
							| 
									
										
										
										
											2022-11-21 07:24:26 -08:00
										 |  |  |     APPVEYOR_BUILD_WORKER_CLOUD: 'electronhq-16-core' | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 15:38:06 +10:00
										 |  |  |   if (!options.ghRelease) { | 
					
						
							| 
									
										
										
										
											2022-05-09 06:34:17 -07:00
										 |  |  |     environmentVariables.UPLOAD_TO_STORAGE = 1; | 
					
						
							| 
									
										
										
										
											2018-05-14 17:21:51 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |   const requestOpts = { | 
					
						
							| 
									
										
										
										
											2019-10-31 14:25:11 -04:00
										 |  |  |     url: BUILD_APPVEYOR_URL, | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |     auth: { | 
					
						
							| 
									
										
										
										
											2019-05-23 16:54:34 -04:00
										 |  |  |       bearer: process.env.APPVEYOR_CLOUD_TOKEN | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |     }, | 
					
						
							|  |  |  |     headers: { | 
					
						
							|  |  |  |       'Content-Type': 'application/json' | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2017-11-03 14:51:40 +08:00
										 |  |  |     body: JSON.stringify({ | 
					
						
							| 
									
										
										
										
											2019-05-23 16:54:34 -04:00
										 |  |  |       accountName: 'electron-bot', | 
					
						
							| 
									
										
										
										
											2018-07-09 15:29:44 -04:00
										 |  |  |       projectSlug: appVeyorJobs[job], | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |       branch: targetBranch, | 
					
						
							| 
									
										
										
										
											2021-06-30 18:17:59 -07:00
										 |  |  |       commitId: options.commit || undefined, | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |       environmentVariables | 
					
						
							|  |  |  |     }), | 
					
						
							|  |  |  |     method: 'POST' | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |   }; | 
					
						
							|  |  |  |   jobRequestedCount++; | 
					
						
							| 
									
										
										
										
											2021-04-13 09:37:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     const { version } = await makeRequest(requestOpts, true); | 
					
						
							|  |  |  |     const buildUrl = `https://ci.appveyor.com/project/electron-bot/${appVeyorJobs[job]}/build/${version}`; | 
					
						
							|  |  |  |     console.log(`AppVeyor release build request for ${job} successful.  Check build status at ${buildUrl}`); | 
					
						
							|  |  |  |   } catch (err) { | 
					
						
							| 
									
										
										
										
											2024-01-12 14:37:39 +01:00
										 |  |  |     if (err.response?.body) { | 
					
						
							|  |  |  |       console.error('Could not call AppVeyor: ', { | 
					
						
							|  |  |  |         statusCode: err.response.statusCode, | 
					
						
							|  |  |  |         body: JSON.parse(err.response.body) | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       console.error('Error calling AppVeyor:', err); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-13 09:37:23 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  | function buildAppVeyor (targetBranch, options) { | 
					
						
							|  |  |  |   const validJobs = Object.keys(appVeyorJobs); | 
					
						
							|  |  |  |   if (options.job) { | 
					
						
							|  |  |  |     assert(validJobs.includes(options.job), `Unknown AppVeyor CI job name: ${options.job}.  Valid values are: ${validJobs}.`); | 
					
						
							|  |  |  |     callAppVeyor(targetBranch, options.job, options); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     for (const job of validJobs) { | 
					
						
							|  |  |  |       callAppVeyor(targetBranch, job, options); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function buildGHActions (targetBranch, options) { | 
					
						
							|  |  |  |   if (options.job) { | 
					
						
							|  |  |  |     assert(ghActionsPublishWorkflows.includes(options.job), `Unknown GitHub Actions workflow name: ${options.job}. Valid values are: ${ghActionsPublishWorkflows}.`); | 
					
						
							|  |  |  |     githubActionsCall(targetBranch, options.job, options); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     assert(!options.arch, 'Cannot provide a single architecture while building all workflows, please specify a single workflow via --workflow'); | 
					
						
							|  |  |  |     options.runningPublishWorkflows = true; | 
					
						
							|  |  |  |     for (const job of ghActionsPublishWorkflows) { | 
					
						
							|  |  |  |       githubActionsCall(targetBranch, job, options); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-05 13:57:47 -04:00
										 |  |  | function runRelease (targetBranch, options) { | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |   if (options.ci) { | 
					
						
							|  |  |  |     switch (options.ci) { | 
					
						
							| 
									
										
										
										
											2024-04-01 07:02:26 -07:00
										 |  |  |       case 'GitHubActions': { | 
					
						
							|  |  |  |         buildGHActions(targetBranch, options); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |       case 'AppVeyor': { | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |         buildAppVeyor(targetBranch, options); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-06-13 16:11:26 -04:00
										 |  |  |       default: { | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |         console.log(`Error! Unknown CI: ${options.ci}.`); | 
					
						
							|  |  |  |         process.exit(1); | 
					
						
							| 
									
										
										
										
											2018-06-13 16:11:26 -04:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |     buildAppVeyor(targetBranch, options); | 
					
						
							| 
									
										
										
										
											2024-07-01 03:33:12 -07:00
										 |  |  |     buildGHActions(targetBranch, options); | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |   console.log(`${jobRequestedCount} jobs were requested.`); | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-11-02 15:26:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  | module.exports = runRelease; | 
					
						
							| 
									
										
										
										
											2017-11-02 15:26:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  | if (require.main === module) { | 
					
						
							| 
									
										
										
										
											2018-05-14 17:21:51 -04:00
										 |  |  |   const args = require('minimist')(process.argv.slice(2), { | 
					
						
							| 
									
										
										
										
											2022-07-28 19:05:59 -04:00
										 |  |  |     boolean: ['ghRelease'] | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |   }); | 
					
						
							|  |  |  |   const targetBranch = args._[0]; | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |   if (args._.length < 1) { | 
					
						
							|  |  |  |     console.log(`Trigger CI to build release builds of electron.
 | 
					
						
							| 
									
										
										
										
											2024-07-01 03:33:12 -07:00
										 |  |  |     Usage: ci-release-build.js [--job=CI_JOB_NAME] [--arch=INDIVIDUAL_ARCH] [--ci=AppVeyor|GitHubActions] | 
					
						
							|  |  |  |     [--ghRelease] [--appveyorJobId=xxx] [--commit=sha] TARGET_BRANCH | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |     `);
 | 
					
						
							|  |  |  |     process.exit(0); | 
					
						
							| 
									
										
										
										
											2017-11-16 16:04:50 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-03-20 13:28:31 -07:00
										 |  |  |   runRelease(targetBranch, args); | 
					
						
							| 
									
										
										
										
											2017-11-03 14:51:40 +08:00
										 |  |  | } |