mirror of
https://github.com/pulsar-edit/pulsar.git
synced 2024-11-10 10:17:11 +03:00
621 lines
19 KiB
JavaScript
621 lines
19 KiB
JavaScript
'use babel'
|
|
|
|
import fs from 'fs-plus'
|
|
import path from 'path'
|
|
import temp from 'temp'
|
|
import Git from 'nodegit'
|
|
|
|
import {it, beforeEach, afterEach} from './async-spec-helpers'
|
|
|
|
import GitRepositoryAsync from '../src/git-repository-async'
|
|
import Project from '../src/project'
|
|
|
|
temp.track()
|
|
|
|
function openFixture (fixture) {
|
|
return GitRepositoryAsync.open(path.join(__dirname, 'fixtures', 'git', fixture))
|
|
}
|
|
|
|
function copyRepository () {
|
|
const workingDirPath = temp.mkdirSync('atom-working-dir')
|
|
fs.copySync(path.join(__dirname, 'fixtures', 'git', 'working-dir'), workingDirPath)
|
|
fs.renameSync(path.join(workingDirPath, 'git.git'), path.join(workingDirPath, '.git'))
|
|
return fs.realpathSync(workingDirPath)
|
|
}
|
|
|
|
describe('GitRepositoryAsync', () => {
|
|
let repo
|
|
|
|
afterEach(() => {
|
|
if (repo != null) repo.destroy()
|
|
})
|
|
|
|
describe('@open(path)', () => {
|
|
it('repo is null when no repository is found', async () => {
|
|
repo = GitRepositoryAsync.open(path.join(temp.dir, 'nogit.txt'))
|
|
|
|
let threw = false
|
|
try {
|
|
await repo.repoPromise
|
|
} catch (e) {
|
|
threw = true
|
|
}
|
|
|
|
expect(threw).toBe(true)
|
|
expect(repo.repo).toBe(null)
|
|
})
|
|
})
|
|
|
|
describe('.getPath()', () => {
|
|
it('returns the repository path for a repository path', async () => {
|
|
repo = openFixture('master.git')
|
|
const repoPath = await repo.getPath()
|
|
expect(repoPath).toBe(path.join(__dirname, 'fixtures', 'git', 'master.git'))
|
|
})
|
|
})
|
|
|
|
describe('.isPathIgnored(path)', () => {
|
|
beforeEach(() => {
|
|
repo = openFixture('ignore.git')
|
|
})
|
|
|
|
it('resolves true for an ignored path', async () => {
|
|
const ignored = await repo.isPathIgnored('a.txt')
|
|
expect(ignored).toBe(true)
|
|
})
|
|
|
|
it('resolves false for a non-ignored path', async () => {
|
|
const ignored = await repo.isPathIgnored('b.txt')
|
|
expect(ignored).toBe(false)
|
|
})
|
|
})
|
|
|
|
describe('.isPathModified(path)', () => {
|
|
let filePath, newPath, emptyPath
|
|
|
|
beforeEach(() => {
|
|
const workingDirPath = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirPath)
|
|
filePath = path.join(workingDirPath, 'a.txt')
|
|
newPath = path.join(workingDirPath, 'new-path.txt')
|
|
fs.writeFileSync(newPath, "i'm new here")
|
|
emptyPath = path.join(workingDirPath, 'empty-path.txt')
|
|
})
|
|
|
|
describe('when the path is unstaged', () => {
|
|
it('resolves false if the path has not been modified', async () => {
|
|
const modified = await repo.isPathModified(filePath)
|
|
expect(modified).toBe(false)
|
|
})
|
|
|
|
it('resolves true if the path is modified', async () => {
|
|
fs.writeFileSync(filePath, 'change')
|
|
const modified = await repo.isPathModified(filePath)
|
|
expect(modified).toBe(true)
|
|
})
|
|
|
|
it('resolves false if the path is new', async () => {
|
|
const modified = await repo.isPathModified(newPath)
|
|
expect(modified).toBe(false)
|
|
})
|
|
|
|
it('resolves false if the path is invalid', async () => {
|
|
const modified = await repo.isPathModified(emptyPath)
|
|
expect(modified).toBe(false)
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('.isPathNew(path)', () => {
|
|
let newPath
|
|
|
|
beforeEach(() => {
|
|
const workingDirPath = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirPath)
|
|
newPath = path.join(workingDirPath, 'new-path.txt')
|
|
fs.writeFileSync(newPath, "i'm new here")
|
|
})
|
|
|
|
describe('when the path is unstaged', () => {
|
|
it('returns true if the path is new', async () => {
|
|
const isNew = await repo.isPathNew(newPath)
|
|
expect(isNew).toBe(true)
|
|
})
|
|
|
|
it("returns false if the path isn't new", async () => {
|
|
const modified = await repo.isPathModified(newPath)
|
|
expect(modified).toBe(false)
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('.checkoutHead(path)', () => {
|
|
let filePath
|
|
|
|
beforeEach(() => {
|
|
const workingDirPath = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirPath)
|
|
filePath = path.join(workingDirPath, 'a.txt')
|
|
})
|
|
|
|
it('no longer reports a path as modified after checkout', async () => {
|
|
let modified = await repo.isPathModified(filePath)
|
|
expect(modified).toBe(false)
|
|
|
|
fs.writeFileSync(filePath, 'ch ch changes')
|
|
|
|
modified = await repo.isPathModified(filePath)
|
|
expect(modified).toBe(true)
|
|
|
|
await repo.checkoutHead(filePath)
|
|
|
|
modified = await repo.isPathModified(filePath)
|
|
expect(modified).toBe(false)
|
|
})
|
|
|
|
it('restores the contents of the path to the original text', async () => {
|
|
fs.writeFileSync(filePath, 'ch ch changes')
|
|
await repo.checkoutHead(filePath)
|
|
expect(fs.readFileSync(filePath, 'utf8')).toBe('')
|
|
})
|
|
|
|
it('fires a did-change-status event if the checkout completes successfully', async () => {
|
|
fs.writeFileSync(filePath, 'ch ch changes')
|
|
|
|
await repo.getPathStatus(filePath)
|
|
|
|
const statusHandler = jasmine.createSpy('statusHandler')
|
|
repo.onDidChangeStatus(statusHandler)
|
|
|
|
await repo.checkoutHead(filePath)
|
|
|
|
expect(statusHandler.callCount).toBe(1)
|
|
expect(statusHandler.argsForCall[0][0]).toEqual({path: filePath, pathStatus: 0})
|
|
|
|
await repo.checkoutHead(filePath)
|
|
expect(statusHandler.callCount).toBe(1)
|
|
})
|
|
})
|
|
|
|
describe('.checkoutHeadForEditor(editor)', () => {
|
|
let filePath
|
|
let editor
|
|
|
|
beforeEach(() => {
|
|
spyOn(atom, 'confirm')
|
|
|
|
const workingDirPath = copyRepository()
|
|
repo = new GitRepositoryAsync(workingDirPath, {project: atom.project, config: atom.config, confirm: atom.confirm})
|
|
filePath = path.join(workingDirPath, 'a.txt')
|
|
fs.writeFileSync(filePath, 'ch ch changes')
|
|
|
|
waitsForPromise(() => atom.workspace.open(filePath))
|
|
runs(() => editor = atom.workspace.getActiveTextEditor())
|
|
})
|
|
|
|
it('displays a confirmation dialog by default', async () => {
|
|
atom.confirm.andCallFake(({buttons}) => buttons.OK())
|
|
atom.config.set('editor.confirmCheckoutHeadRevision', true)
|
|
|
|
await repo.checkoutHeadForEditor(editor)
|
|
|
|
expect(fs.readFileSync(filePath, 'utf8')).toBe('')
|
|
})
|
|
|
|
it('does not display a dialog when confirmation is disabled', async () => {
|
|
atom.config.set('editor.confirmCheckoutHeadRevision', false)
|
|
|
|
await repo.checkoutHeadForEditor(editor)
|
|
|
|
expect(fs.readFileSync(filePath, 'utf8')).toBe('')
|
|
expect(atom.confirm).not.toHaveBeenCalled()
|
|
})
|
|
})
|
|
|
|
describe('.destroy()', () => {
|
|
it('throws an exception when any method is called after it is called', async () => {
|
|
repo = new GitRepositoryAsync(require.resolve('./fixtures/git/master.git/HEAD'))
|
|
repo.destroy()
|
|
|
|
let error = null
|
|
try {
|
|
await repo.getShortHead()
|
|
} catch (e) {
|
|
error = e
|
|
}
|
|
expect(error.name).toBe(GitRepositoryAsync.DestroyedErrorName)
|
|
})
|
|
})
|
|
|
|
describe('.getPathStatus(path)', () => {
|
|
let filePath
|
|
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
filePath = path.join(workingDirectory, 'file.txt')
|
|
})
|
|
|
|
it('trigger a status-changed event when the new status differs from the last cached one', async () => {
|
|
const statusHandler = jasmine.createSpy('statusHandler')
|
|
repo.onDidChangeStatus(statusHandler)
|
|
fs.writeFileSync(filePath, '')
|
|
|
|
await repo.getPathStatus(filePath)
|
|
|
|
expect(statusHandler.callCount).toBe(1)
|
|
const status = Git.Status.STATUS.WT_MODIFIED
|
|
expect(statusHandler.argsForCall[0][0]).toEqual({path: filePath, pathStatus: status})
|
|
fs.writeFileSync(filePath, 'abc')
|
|
|
|
await repo.getPathStatus(filePath)
|
|
expect(statusHandler.callCount).toBe(1)
|
|
})
|
|
})
|
|
|
|
describe('.getDirectoryStatus(path)', () => {
|
|
let directoryPath, filePath
|
|
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
directoryPath = path.join(workingDirectory, 'dir')
|
|
filePath = path.join(directoryPath, 'b.txt')
|
|
})
|
|
|
|
it('gets the status based on the files inside the directory', async () => {
|
|
await repo.checkoutHead(filePath)
|
|
|
|
let result = await repo.getDirectoryStatus(directoryPath)
|
|
expect(repo.isStatusModified(result)).toBe(false)
|
|
|
|
fs.writeFileSync(filePath, 'abc')
|
|
|
|
result = await repo.getDirectoryStatus(directoryPath)
|
|
expect(repo.isStatusModified(result)).toBe(true)
|
|
})
|
|
})
|
|
|
|
describe('.refreshStatus()', () => {
|
|
let newPath, modifiedPath, cleanPath
|
|
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
modifiedPath = path.join(workingDirectory, 'file.txt')
|
|
newPath = path.join(workingDirectory, 'untracked.txt')
|
|
cleanPath = path.join(workingDirectory, 'other.txt')
|
|
fs.writeFileSync(cleanPath, 'Full of text')
|
|
fs.writeFileSync(newPath, '')
|
|
newPath = fs.absolute(newPath) // specs could be running under symbol path.
|
|
})
|
|
|
|
it('returns status information for all new and modified files', async () => {
|
|
fs.writeFileSync(modifiedPath, 'making this path modified')
|
|
await repo.refreshStatus()
|
|
|
|
expect(await repo.getCachedPathStatus(cleanPath)).toBeUndefined()
|
|
expect(repo.isStatusNew(await repo.getCachedPathStatus(newPath))).toBe(true)
|
|
expect(repo.isStatusModified(await repo.getCachedPathStatus(modifiedPath))).toBe(true)
|
|
})
|
|
})
|
|
|
|
describe('.isProjectAtRoot()', () => {
|
|
it('returns true when the repository is at the root', async () => {
|
|
const workingDirectory = copyRepository()
|
|
atom.project.setPaths([workingDirectory])
|
|
const repo = atom.project.getRepositories()[0].async
|
|
|
|
const atRoot = await repo.isProjectAtRoot()
|
|
expect(atRoot).toBe(true)
|
|
})
|
|
|
|
it("returns false when the repository wasn't created with a project", async () => {
|
|
const workingDirectory = copyRepository()
|
|
const repo = GitRepositoryAsync.open(workingDirectory)
|
|
|
|
const atRoot = await repo.isProjectAtRoot()
|
|
expect(atRoot).toBe(false)
|
|
})
|
|
})
|
|
|
|
describe('buffer events', () => {
|
|
let repo
|
|
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
atom.project.setPaths([workingDirectory])
|
|
|
|
// When the path is added to the project, the repository is refreshed. We
|
|
// need to wait for that to complete before the tests continue so that
|
|
// we're in a known state.
|
|
repo = atom.project.getRepositories()[0].async
|
|
waitsFor(() => !repo._isRefreshing())
|
|
})
|
|
|
|
it('emits a status-changed event when a buffer is saved', async () => {
|
|
const editor = await atom.workspace.open('other.txt')
|
|
|
|
editor.insertNewline()
|
|
|
|
const statusHandler = jasmine.createSpy('statusHandler')
|
|
repo.onDidChangeStatus(statusHandler)
|
|
editor.save()
|
|
|
|
waitsFor(() => statusHandler.callCount > 0)
|
|
runs(() => {
|
|
expect(statusHandler.callCount).toBeGreaterThan(0)
|
|
expect(statusHandler).toHaveBeenCalledWith({path: editor.getPath(), pathStatus: 256})
|
|
})
|
|
})
|
|
|
|
it('emits a status-changed event when a buffer is reloaded', async () => {
|
|
const editor = await atom.workspace.open('other.txt')
|
|
|
|
fs.writeFileSync(editor.getPath(), 'changed')
|
|
|
|
const statusHandler = jasmine.createSpy('statusHandler')
|
|
repo.onDidChangeStatus(statusHandler)
|
|
editor.getBuffer().reload()
|
|
|
|
waitsFor(() => statusHandler.callCount > 0)
|
|
runs(() => {
|
|
expect(statusHandler.callCount).toBeGreaterThan(0)
|
|
expect(statusHandler).toHaveBeenCalledWith({path: editor.getPath(), pathStatus: 256})
|
|
})
|
|
})
|
|
|
|
it("emits a status-changed event when a buffer's path changes", async () => {
|
|
const editor = await atom.workspace.open('other.txt')
|
|
|
|
fs.writeFileSync(editor.getPath(), 'changed')
|
|
|
|
const statusHandler = jasmine.createSpy('statusHandler')
|
|
repo.onDidChangeStatus(statusHandler)
|
|
editor.getBuffer().emitter.emit('did-change-path')
|
|
|
|
waitsFor(() => statusHandler.callCount > 0)
|
|
runs(() => {
|
|
expect(statusHandler.callCount).toBeGreaterThan(0)
|
|
expect(statusHandler).toHaveBeenCalledWith({path: editor.getPath(), pathStatus: 256})
|
|
|
|
const pathHandler = jasmine.createSpy('pathHandler')
|
|
const buffer = editor.getBuffer()
|
|
buffer.onDidChangePath(pathHandler)
|
|
buffer.emitter.emit('did-change-path')
|
|
|
|
waitsFor(() => pathHandler.callCount > 0)
|
|
runs(() => expect(pathHandler.callCount).toBeGreaterThan(0))
|
|
})
|
|
})
|
|
|
|
it('stops listening to the buffer when the repository is destroyed (regression)', async () => {
|
|
const editor = await atom.workspace.open('other.txt')
|
|
const repo = atom.project.getRepositories()[0]
|
|
repo.destroy()
|
|
expect(() => editor.save()).not.toThrow()
|
|
})
|
|
})
|
|
|
|
describe('when a project is deserialized', () => {
|
|
let project2
|
|
|
|
beforeEach(() => {
|
|
atom.project.setPaths([copyRepository()])
|
|
|
|
// See the comment in the 'buffer events' beforeEach for why we need to do
|
|
// this.
|
|
const repository = atom.project.getRepositories()[0].async
|
|
waitsFor(() => !repository._isRefreshing())
|
|
})
|
|
|
|
afterEach(() => {
|
|
if (project2) project2.destroy()
|
|
})
|
|
|
|
it('subscribes to all the serialized buffers in the project', async () => {
|
|
await atom.workspace.open('file.txt')
|
|
|
|
project2 = new Project({notificationManager: atom.notifications, packageManager: atom.packages, confirm: atom.confirm})
|
|
project2.deserialize(atom.project.serialize(), atom.deserializers)
|
|
|
|
const repo = project2.getRepositories()[0].async
|
|
waitsFor(() => !repo._isRefreshing())
|
|
runs(() => {
|
|
const buffer = project2.getBuffers()[0]
|
|
|
|
waitsFor(() => buffer.loaded)
|
|
runs(() => {
|
|
buffer.append('changes')
|
|
|
|
const statusHandler = jasmine.createSpy('statusHandler')
|
|
repo.onDidChangeStatus(statusHandler)
|
|
buffer.save()
|
|
|
|
waitsFor(() => statusHandler.callCount > 0)
|
|
runs(() => {
|
|
expect(statusHandler.callCount).toBeGreaterThan(0)
|
|
expect(statusHandler).toHaveBeenCalledWith({path: buffer.getPath(), pathStatus: 256})
|
|
})
|
|
})
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('GitRepositoryAsync::relativize(filePath, workdir)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
// This is a change in implementation from the git-utils version
|
|
it('just returns path if workdir is not provided', () => {
|
|
const _path = '/foo/bar/baz.txt'
|
|
const relPath = repo.relativize(_path)
|
|
expect(_path).toEqual(relPath)
|
|
})
|
|
|
|
it('relativizes a repo path', () => {
|
|
const workdir = '/tmp/foo/bar/baz/'
|
|
const relativizedPath = repo.relativize(`${workdir}a/b.txt`, workdir)
|
|
expect(relativizedPath).toBe('a/b.txt')
|
|
})
|
|
|
|
it("doesn't require workdir to end in a slash", () => {
|
|
const workdir = '/tmp/foo/bar/baz'
|
|
const relativizedPath = repo.relativize(`${workdir}/a/b.txt`, workdir)
|
|
expect(relativizedPath).toBe('a/b.txt')
|
|
})
|
|
})
|
|
|
|
describe('.getShortHead(path)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('returns the human-readable branch name', async () => {
|
|
const head = await repo.getShortHead()
|
|
expect(head).toBe('master')
|
|
})
|
|
})
|
|
|
|
describe('.getAheadBehindCount(reference, path)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('returns 0, 0 for a branch with no upstream', async () => {
|
|
const {ahead, behind} = await repo.getAheadBehindCount('master')
|
|
expect(ahead).toBe(0)
|
|
expect(behind).toBe(0)
|
|
})
|
|
})
|
|
|
|
describe('.getCachedUpstreamAheadBehindCount(path)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('returns 0, 0 for a branch with no upstream', async () => {
|
|
await repo.refreshStatus()
|
|
const {ahead, behind} = repo.getCachedUpstreamAheadBehindCount()
|
|
expect(ahead).toBe(0)
|
|
expect(behind).toBe(0)
|
|
})
|
|
})
|
|
|
|
describe('.getDiffStats(path)', () => {
|
|
let workingDirectory
|
|
beforeEach(() => {
|
|
workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('returns the diff stat', async () => {
|
|
const filePath = path.join(workingDirectory, 'a.txt')
|
|
fs.writeFileSync(filePath, 'change')
|
|
|
|
const {added, deleted} = await repo.getDiffStats('a.txt')
|
|
expect(added).toBe(1)
|
|
expect(deleted).toBe(0)
|
|
})
|
|
})
|
|
|
|
describe('.hasBranch(branch)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('resolves true when the branch exists', async () => {
|
|
const hasBranch = await repo.hasBranch('master')
|
|
expect(hasBranch).toBe(true)
|
|
})
|
|
|
|
it("resolves false when the branch doesn't exist", async () => {
|
|
const hasBranch = await repo.hasBranch('trolleybus')
|
|
expect(hasBranch).toBe(false)
|
|
})
|
|
})
|
|
|
|
describe('.getReferences(path)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('returns the heads, remotes, and tags', async () => {
|
|
const {heads, remotes, tags} = await repo.getReferences()
|
|
expect(heads.length).toBe(1)
|
|
expect(remotes.length).toBe(0)
|
|
expect(tags.length).toBe(0)
|
|
})
|
|
})
|
|
|
|
describe('.getReferenceTarget(reference, path)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('returns the SHA target', async () => {
|
|
const SHA = await repo.getReferenceTarget('refs/heads/master')
|
|
expect(SHA).toBe('8a9c86f1cb1f14b8f436eb91f4b052c8802ca99e')
|
|
})
|
|
})
|
|
|
|
describe('.getConfigValue(key, path)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('looks up the value for the key', async () => {
|
|
const bare = await repo.getConfigValue('core.bare')
|
|
expect(bare).toBe('false')
|
|
})
|
|
|
|
it("resolves to null if there's no value", async () => {
|
|
const value = await repo.getConfigValue('my.special.key')
|
|
expect(value).toBeNull()
|
|
})
|
|
})
|
|
|
|
describe('.checkoutReference(reference, create)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('can create new branches', () => {
|
|
let success = false
|
|
let threw = false
|
|
waitsForPromise(() => repo.checkoutReference('my-b', true)
|
|
.then(_ => success = true)
|
|
.catch(_ => threw = true))
|
|
runs(() => {
|
|
expect(success).toBe(true)
|
|
expect(threw).toBe(false)
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('.getLineDiffs(path, text)', () => {
|
|
beforeEach(() => {
|
|
const workingDirectory = copyRepository()
|
|
repo = GitRepositoryAsync.open(workingDirectory)
|
|
})
|
|
|
|
it('returns the old and new lines of the diff', async () => {
|
|
const {oldStart, newStart, oldLines, newLines} = await repo.getLineDiffs('a.txt', 'hi there')
|
|
expect(oldStart).toBe(0)
|
|
expect(oldLines).toBe(0)
|
|
expect(newStart).toBe(1)
|
|
expect(newLines).toBe(1)
|
|
})
|
|
})
|
|
})
|