mirror of
https://github.com/serverless/serverless.git
synced 2026-02-01 16:07:28 +00:00
480 lines
15 KiB
JavaScript
480 lines
15 KiB
JavaScript
'use strict'
|
|
|
|
const sinon = require('sinon')
|
|
const fse = require('fs-extra')
|
|
const path = require('path')
|
|
const os = require('os')
|
|
const proxyquire = require('proxyquire')
|
|
const chai = require('chai')
|
|
|
|
const { expect } = chai
|
|
const { getTmpDirPath } = require('../../../utils/fs')
|
|
|
|
const writeFileSync = require('../../../../lib/utils/fs/write-file-sync')
|
|
const readFileSync = require('../../../../lib/utils/fs/read-file-sync')
|
|
|
|
chai.use(require('chai-as-promised'))
|
|
|
|
describe('downloadTemplateFromRepo', () => {
|
|
let downloadTemplateFromRepo
|
|
let spawnStub
|
|
let downloadStub
|
|
let cwd
|
|
|
|
let parseRepoURL
|
|
let fetchStub
|
|
|
|
let serviceDir
|
|
let newServicePath
|
|
|
|
beforeEach(() => {
|
|
const tmpDir = getTmpDirPath()
|
|
cwd = process.cwd()
|
|
|
|
fse.mkdirsSync(tmpDir)
|
|
process.chdir(tmpDir)
|
|
|
|
serviceDir = tmpDir
|
|
newServicePath = path.join(serviceDir, 'new-service-name')
|
|
|
|
fetchStub = sinon.stub().resolves({
|
|
json: () => ({
|
|
displayName: 'Bitbucket',
|
|
}),
|
|
})
|
|
|
|
spawnStub = sinon.stub().resolves()
|
|
downloadStub = sinon.stub().resolves()
|
|
|
|
const downloadTemplateFromRepoModule = proxyquire(
|
|
'../../../../lib/utils/download-template-from-repo',
|
|
{
|
|
'node-fetch': async (url) => {
|
|
if (url.indexOf('mybitbucket.server.ltd') > -1) {
|
|
return fetchStub()
|
|
}
|
|
|
|
throw Error('unknown server type')
|
|
},
|
|
'@serverless/utils/download': downloadStub,
|
|
'child-process-ext/spawn': spawnStub,
|
|
},
|
|
)
|
|
downloadTemplateFromRepo =
|
|
downloadTemplateFromRepoModule.downloadTemplateFromRepo
|
|
parseRepoURL = downloadTemplateFromRepoModule.parseRepoURL
|
|
})
|
|
|
|
afterEach((done) => {
|
|
// change back to the old cwd
|
|
process.chdir(cwd)
|
|
fse.remove(newServicePath).then(done)
|
|
})
|
|
|
|
describe('downloadTemplateFromRepo', () => {
|
|
it('should reject an error if the passed URL option is not a valid URL', () => {
|
|
return expect(
|
|
downloadTemplateFromRepo('invalidUrl'),
|
|
).to.be.eventually.rejected.and.have.property(
|
|
'code',
|
|
'INVALID_TEMPLATE_URL',
|
|
)
|
|
})
|
|
|
|
it('should reject an error if the passed URL is not a valid GitHub URL', () => {
|
|
return expect(
|
|
downloadTemplateFromRepo('http://no-git-hub-url.com/foo/bar'),
|
|
).to.be.eventually.rejected.and.have.property(
|
|
'code',
|
|
'INVALID_TEMPLATE_PROVIDER',
|
|
)
|
|
})
|
|
|
|
it('should reject an error if a directory with the same service name is already present', () => {
|
|
const serviceDirName = path.join(serviceDir, 'existing-service')
|
|
fse.mkdirsSync(serviceDirName)
|
|
|
|
return expect(
|
|
downloadTemplateFromRepo('https://github.com/johndoe/existing-service'),
|
|
).to.be.eventually.rejected.and.have.property(
|
|
'code',
|
|
'TARGET_FOLDER_ALREADY_EXISTS',
|
|
)
|
|
})
|
|
|
|
it('should download the service based on a regular .git URL', async () => {
|
|
const url = 'https://example.com/sample-service.git'
|
|
|
|
return expect(downloadTemplateFromRepo(url)).to.be.fulfilled.then(() => {
|
|
expect(spawnStub.calledOnce).to.equal(true)
|
|
expect(downloadStub.calledOnce).to.equal(false)
|
|
expect(spawnStub.args[0][0]).to.equal('git')
|
|
expect(spawnStub.args[0][1][0]).to.equal('clone')
|
|
expect(spawnStub.args[0][1][1]).to.equal(url)
|
|
})
|
|
})
|
|
|
|
it('should download and rename the service based on a regular .git URL', async () => {
|
|
const url = 'https://example.com/sample-service.git'
|
|
const name = 'new-service-name'
|
|
|
|
spawnStub.resolves({
|
|
then: (callback) => {
|
|
const slsYml = path.join(
|
|
process.cwd(),
|
|
'new-service-name',
|
|
'serverless.yml',
|
|
)
|
|
writeFileSync(slsYml, 'service: sample-service')
|
|
callback()
|
|
},
|
|
})
|
|
|
|
return expect(downloadTemplateFromRepo(url, name)).to.be.fulfilled.then(
|
|
(serviceName) => {
|
|
expect(spawnStub.calledOnce).to.equal(true)
|
|
expect(downloadStub.calledOnce).to.equal(false)
|
|
expect(spawnStub.args[0][0]).to.equal('git')
|
|
expect(spawnStub.args[0][1][0]).to.equal('clone')
|
|
expect(spawnStub.args[0][1][1]).to.equal(url)
|
|
const yml = readFileSync(path.join(newServicePath, 'serverless.yml'))
|
|
expect(yml.service).to.equal(name)
|
|
expect(serviceName).to.equal('sample-service')
|
|
},
|
|
)
|
|
})
|
|
|
|
it('should download the service based on a regular .git URL start with git@', async () => {
|
|
const url = 'git@example.com/sample-service.git'
|
|
|
|
return expect(downloadTemplateFromRepo(url)).to.be.fulfilled.then(() => {
|
|
expect(spawnStub.calledOnce).to.equal(true)
|
|
expect(downloadStub.calledOnce).to.equal(false)
|
|
expect(spawnStub.args[0][0]).to.equal('git')
|
|
expect(spawnStub.args[0][1][0]).to.equal('clone')
|
|
expect(spawnStub.args[0][1][1]).to.equal(url)
|
|
})
|
|
})
|
|
|
|
it('should download and rename the service based on a regular .git URL start with git@', async () => {
|
|
const url = 'git@example.com/sample-service.git'
|
|
const name = 'new-service-name'
|
|
|
|
spawnStub.resolves({
|
|
then: (callback) => {
|
|
const slsYml = path.join(
|
|
process.cwd(),
|
|
'new-service-name',
|
|
'serverless.yml',
|
|
)
|
|
writeFileSync(slsYml, 'service: sample-service')
|
|
callback()
|
|
},
|
|
})
|
|
|
|
return expect(downloadTemplateFromRepo(url, name)).to.be.fulfilled.then(
|
|
(serviceName) => {
|
|
expect(spawnStub.calledOnce).to.equal(true)
|
|
expect(downloadStub.calledOnce).to.equal(false)
|
|
expect(spawnStub.args[0][0]).to.equal('git')
|
|
expect(spawnStub.args[0][1][0]).to.equal('clone')
|
|
expect(spawnStub.args[0][1][1]).to.equal(url)
|
|
const yml = readFileSync(path.join(newServicePath, 'serverless.yml'))
|
|
expect(yml.service).to.equal(name)
|
|
expect(serviceName).to.equal('sample-service')
|
|
},
|
|
)
|
|
})
|
|
|
|
it('should download the service based on the GitHub URL', async () => {
|
|
const url = 'https://github.com/johndoe/service-to-be-downloaded'
|
|
|
|
return expect(downloadTemplateFromRepo(url)).to.be.fulfilled.then(() => {
|
|
expect(downloadStub.calledOnce).to.equal(true)
|
|
expect(downloadStub.args[0][0]).to.equal(`${url}/archive/master.zip`)
|
|
})
|
|
})
|
|
|
|
it('should download and rename the service based on the GitHub URL', async () => {
|
|
const url = 'https://github.com/johndoe/service-to-be-downloaded'
|
|
const name = 'new-service-name'
|
|
|
|
downloadStub.resolves({
|
|
then: (callback) => {
|
|
const slsYml = path.join(
|
|
process.cwd(),
|
|
'new-service-name',
|
|
'serverless.yml',
|
|
)
|
|
writeFileSync(slsYml, 'service: service-name')
|
|
callback()
|
|
},
|
|
})
|
|
|
|
return expect(downloadTemplateFromRepo(url, name)).to.be.fulfilled.then(
|
|
(serviceName) => {
|
|
expect(downloadStub.calledOnce).to.equal(true)
|
|
expect(downloadStub.args[0][1]).to.contain(name)
|
|
expect(downloadStub.args[0][0]).to.equal(`${url}/archive/master.zip`)
|
|
const yml = readFileSync(path.join(newServicePath, 'serverless.yml'))
|
|
expect(yml.service).to.equal(name)
|
|
expect(serviceName).to.equal('service-to-be-downloaded')
|
|
},
|
|
)
|
|
})
|
|
|
|
it('should download and rename the service based directories in the GitHub URL', async () => {
|
|
const url =
|
|
'https://github.com/serverless/examples/tree/master/rest-api-with-dynamodb'
|
|
const name = 'new-service-name'
|
|
|
|
downloadStub.resolves(
|
|
fse.remove(newServicePath).then(() => {
|
|
const slsYml = path.join(
|
|
os.tmpdir(),
|
|
'examples',
|
|
'rest-api-with-dynamodb',
|
|
'serverless.yml',
|
|
)
|
|
writeFileSync(slsYml, 'service: service-name')
|
|
}),
|
|
)
|
|
|
|
return expect(downloadTemplateFromRepo(url, name)).to.be.fulfilled.then(
|
|
(serviceName) => {
|
|
expect(downloadStub.calledOnce).to.equal(true)
|
|
const yml = readFileSync(path.join(newServicePath, 'serverless.yml'))
|
|
expect(yml.service).to.equal(name)
|
|
expect(serviceName).to.equal('rest-api-with-dynamodb')
|
|
},
|
|
)
|
|
})
|
|
|
|
it('should throw an error if the same service name exists as directory in Github', () => {
|
|
const url =
|
|
'https://github.com/serverless/examples/tree/master/rest-api-with-dynamodb'
|
|
const serviceDirName = path.join(serviceDir, 'rest-api-with-dynamodb')
|
|
fse.mkdirsSync(serviceDirName)
|
|
|
|
return expect(
|
|
downloadTemplateFromRepo(null, url),
|
|
).to.be.eventually.rejected.and.have.property(
|
|
'code',
|
|
'MISSING_TEMPLATE_URL',
|
|
)
|
|
})
|
|
})
|
|
|
|
describe('parseRepoURL', () => {
|
|
it('should reject an error if no URL is provided', () => {
|
|
return expect(parseRepoURL()).to.be.eventually.rejected.and.have.property(
|
|
'code',
|
|
'MISSING_TEMPLATE_URL',
|
|
)
|
|
})
|
|
|
|
it('should reject an error if URL is not valid', () => {
|
|
return expect(
|
|
parseRepoURL('non_valid_url'),
|
|
).to.be.eventually.rejected.and.have.property(
|
|
'code',
|
|
'INVALID_TEMPLATE_URL',
|
|
)
|
|
})
|
|
|
|
it('should throw an error if URL is not of valid provider', () => {
|
|
return expect(
|
|
parseRepoURL('https://kostasbariotis.com/repo/owner'),
|
|
).to.be.eventually.rejected.and.have.property(
|
|
'code',
|
|
'INVALID_TEMPLATE_PROVIDER',
|
|
)
|
|
})
|
|
|
|
it('should parse a valid GitHub URL', async () => {
|
|
return expect(
|
|
parseRepoURL('https://github.com/serverless/serverless'),
|
|
).to.be.fulfilled.then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'serverless',
|
|
repo: 'serverless',
|
|
branch: 'master',
|
|
downloadUrl:
|
|
'https://github.com/serverless/serverless/archive/master.zip',
|
|
isSubdirectory: false,
|
|
pathToDirectory: '',
|
|
username: '',
|
|
password: '',
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should parse a valid GitHub URL with subdirectory', async () => {
|
|
return expect(
|
|
parseRepoURL(
|
|
'https://github.com/serverless/serverless/tree/master/assets',
|
|
),
|
|
).to.be.fulfilled.then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'serverless',
|
|
repo: 'serverless',
|
|
branch: 'master',
|
|
downloadUrl:
|
|
'https://github.com/serverless/serverless/archive/master.zip',
|
|
isSubdirectory: true,
|
|
pathToDirectory: 'assets',
|
|
username: '',
|
|
password: '',
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should parse a valid GitHub Entreprise URL', async () => {
|
|
return expect(
|
|
parseRepoURL('https://github.mydomain.com/serverless/serverless'),
|
|
).to.be.fulfilled.then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'serverless',
|
|
repo: 'serverless',
|
|
branch: 'master',
|
|
downloadUrl:
|
|
'https://github.mydomain.com/serverless/serverless/archive/master.zip',
|
|
isSubdirectory: false,
|
|
pathToDirectory: '',
|
|
username: '',
|
|
password: '',
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should parse a valid GitHub Entreprise with subdirectory', async () => {
|
|
return expect(
|
|
parseRepoURL(
|
|
'https://github.mydomain.com/serverless/serverless/tree/master/assets',
|
|
),
|
|
).to.be.fulfilled.then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'serverless',
|
|
repo: 'serverless',
|
|
branch: 'master',
|
|
downloadUrl:
|
|
'https://github.mydomain.com/serverless/serverless/archive/master.zip',
|
|
isSubdirectory: true,
|
|
pathToDirectory: 'assets',
|
|
username: '',
|
|
password: '',
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should parse a valid GitHub Entreprise URL with authentication', async () => {
|
|
return expect(
|
|
parseRepoURL(
|
|
'https://username:password@github.com/serverless/serverless/',
|
|
),
|
|
).to.be.fulfilled.then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'serverless',
|
|
repo: 'serverless',
|
|
branch: 'master',
|
|
downloadUrl:
|
|
'https://github.com/serverless/serverless/archive/master.zip',
|
|
isSubdirectory: false,
|
|
username: 'username',
|
|
password: 'password',
|
|
pathToDirectory: '',
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should parse a valid BitBucket URL', async () => {
|
|
return parseRepoURL('https://bitbucket.org/atlassian/localstack').then(
|
|
(output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'atlassian',
|
|
repo: 'localstack',
|
|
branch: 'master',
|
|
downloadUrl:
|
|
'https://bitbucket.org/atlassian/localstack/get/master.zip',
|
|
isSubdirectory: false,
|
|
pathToDirectory: '',
|
|
username: '',
|
|
password: '',
|
|
})
|
|
},
|
|
)
|
|
})
|
|
|
|
it('should parse a valid BitBucket URL with subdirectory', async () => {
|
|
return parseRepoURL(
|
|
'https://bitbucket.org/atlassian/localstack/src/85870856fd6941ae75c0fa946a51cf756ff2f53a/localstack/dashboard/?at=mvn',
|
|
).then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'atlassian',
|
|
repo: 'localstack',
|
|
branch: 'mvn',
|
|
downloadUrl: 'https://bitbucket.org/atlassian/localstack/get/mvn.zip',
|
|
isSubdirectory: true,
|
|
pathToDirectory: `localstack${path.sep}dashboard`,
|
|
username: '',
|
|
password: '',
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should parse a valid Bitbucket Server URL', async () => {
|
|
return parseRepoURL(
|
|
'https://user:pass@mybitbucket.server.ltd/rest/api/latest/projects/myproject/repos/myrepo/archive?at=refs%2Fheads%2Fdevelop',
|
|
).then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'myproject',
|
|
repo: 'myrepo',
|
|
branch: 'refs/heads/develop',
|
|
downloadUrl:
|
|
'https://mybitbucket.server.ltd/rest/api/latest/projects/myproject/repos/myrepo/archive?at=refs%2Fheads%2Fdevelop&format=zip',
|
|
isSubdirectory: false,
|
|
pathToDirectory: '',
|
|
username: 'user',
|
|
password: 'pass',
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should parse a valid GitLab URL ', async () => {
|
|
return parseRepoURL('https://gitlab.com/serverless/serverless').then(
|
|
(output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'serverless',
|
|
repo: 'serverless',
|
|
branch: 'master',
|
|
downloadUrl:
|
|
'https://gitlab.com/serverless/serverless/-/archive/master/serverless-master.zip',
|
|
isSubdirectory: false,
|
|
pathToDirectory: '',
|
|
username: '',
|
|
password: '',
|
|
})
|
|
},
|
|
)
|
|
})
|
|
|
|
it('should parse a valid GitLab URL with subdirectory', async () => {
|
|
return parseRepoURL(
|
|
'https://gitlab.com/serverless/serverless/tree/dev/subdir',
|
|
).then((output) => {
|
|
expect(output).to.deep.eq({
|
|
owner: 'serverless',
|
|
repo: 'serverless',
|
|
branch: 'dev',
|
|
downloadUrl:
|
|
'https://gitlab.com/serverless/serverless/-/archive/dev/serverless-dev.zip',
|
|
isSubdirectory: true,
|
|
pathToDirectory: 'subdir',
|
|
username: '',
|
|
password: '',
|
|
})
|
|
})
|
|
})
|
|
})
|
|
})
|