2013-03-03 22:43:52 -05:00
|
|
|
require "spec_helper"
|
|
|
|
|
2017-07-10 10:24:02 -04:00
|
|
|
describe WikiPage do
|
2017-08-02 15:55:11 -04:00
|
|
|
let(:project) { create(:project) }
|
2013-03-03 22:43:52 -05:00
|
|
|
let(:user) { project.owner }
|
2014-04-09 07:35:58 -04:00
|
|
|
let(:wiki) { ProjectWiki.new(project, user) }
|
2013-03-03 22:43:52 -05:00
|
|
|
|
2017-07-25 13:09:00 -04:00
|
|
|
subject { described_class.new(wiki) }
|
2013-03-03 22:43:52 -05:00
|
|
|
|
2016-12-17 13:38:26 -05:00
|
|
|
describe '.group_by_directory' do
|
2016-12-15 23:12:21 -05:00
|
|
|
context 'when there are no pages' do
|
2016-12-26 17:12:15 -05:00
|
|
|
it 'returns an empty array' do
|
2017-07-25 13:09:00 -04:00
|
|
|
expect(described_class.group_by_directory(nil)).to eq([])
|
|
|
|
expect(described_class.group_by_directory([])).to eq([])
|
2016-12-15 23:12:21 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when there are pages' do
|
2016-12-17 13:38:26 -05:00
|
|
|
before do
|
2016-12-18 23:34:35 -05:00
|
|
|
create_page('dir_1/dir_1_1/page_3', 'content')
|
2016-12-17 13:38:26 -05:00
|
|
|
create_page('dir_1/page_2', 'content')
|
2016-12-18 23:34:35 -05:00
|
|
|
create_page('dir_2/page_5', 'content')
|
|
|
|
create_page('dir_2/page_4', 'content')
|
|
|
|
create_page('page_1', 'content')
|
2016-12-17 13:38:26 -05:00
|
|
|
end
|
2016-12-26 17:12:15 -05:00
|
|
|
let(:page_1) { wiki.find_page('page_1') }
|
|
|
|
let(:dir_1) do
|
|
|
|
WikiDirectory.new('dir_1', [wiki.find_page('dir_1/page_2')])
|
|
|
|
end
|
|
|
|
let(:dir_1_1) do
|
|
|
|
WikiDirectory.new('dir_1/dir_1_1', [wiki.find_page('dir_1/dir_1_1/page_3')])
|
|
|
|
end
|
|
|
|
let(:dir_2) do
|
|
|
|
pages = [wiki.find_page('dir_2/page_5'),
|
|
|
|
wiki.find_page('dir_2/page_4')]
|
|
|
|
WikiDirectory.new('dir_2', pages)
|
|
|
|
end
|
2016-12-17 13:38:26 -05:00
|
|
|
|
2016-12-26 17:12:15 -05:00
|
|
|
it 'returns an array with pages and directories' do
|
|
|
|
expected_grouped_entries = [page_1, dir_1, dir_1_1, dir_2]
|
2016-12-17 13:38:26 -05:00
|
|
|
|
2017-07-25 13:09:00 -04:00
|
|
|
grouped_entries = described_class.group_by_directory(wiki.pages)
|
2016-12-17 13:38:26 -05:00
|
|
|
|
2016-12-26 17:12:15 -05:00
|
|
|
grouped_entries.each_with_index do |page_or_dir, i|
|
|
|
|
expected_page_or_dir = expected_grouped_entries[i]
|
|
|
|
expected_slugs = get_slugs(expected_page_or_dir)
|
|
|
|
slugs = get_slugs(page_or_dir)
|
2016-12-15 23:12:21 -05:00
|
|
|
|
2016-12-17 13:38:26 -05:00
|
|
|
expect(slugs).to match_array(expected_slugs)
|
|
|
|
end
|
2016-12-15 23:12:21 -05:00
|
|
|
end
|
2016-12-18 23:34:35 -05:00
|
|
|
|
2016-12-26 17:12:15 -05:00
|
|
|
it 'returns an array sorted by alphabetical position' do
|
|
|
|
# Directories and pages within directories are sorted alphabetically.
|
|
|
|
# Pages at root come before everything.
|
|
|
|
expected_order = ['page_1', 'dir_1/page_2', 'dir_1/dir_1_1/page_3',
|
|
|
|
'dir_2/page_4', 'dir_2/page_5']
|
2016-12-18 23:34:35 -05:00
|
|
|
|
2017-07-25 13:09:00 -04:00
|
|
|
grouped_entries = described_class.group_by_directory(wiki.pages)
|
2016-12-18 23:34:35 -05:00
|
|
|
|
2016-12-26 17:12:15 -05:00
|
|
|
actual_order =
|
|
|
|
grouped_entries.map do |page_or_dir|
|
|
|
|
get_slugs(page_or_dir)
|
2017-06-21 09:48:12 -04:00
|
|
|
end
|
|
|
|
.flatten
|
2016-12-26 17:12:15 -05:00
|
|
|
expect(actual_order).to eq(expected_order)
|
2016-12-18 23:34:35 -05:00
|
|
|
end
|
2016-12-15 23:12:21 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-01-29 22:07:31 -05:00
|
|
|
describe '.unhyphenize' do
|
|
|
|
it 'removes hyphens from a name' do
|
|
|
|
name = 'a-name--with-hyphens'
|
|
|
|
|
2017-07-25 13:09:00 -04:00
|
|
|
expect(described_class.unhyphenize(name)).to eq('a name with hyphens')
|
2017-01-29 22:07:31 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-03-03 22:43:52 -05:00
|
|
|
describe "#initialize" do
|
|
|
|
context "when initialized with an existing gollum page" do
|
|
|
|
before do
|
|
|
|
create_page("test page", "test content")
|
|
|
|
@page = wiki.wiki.paged("test page")
|
2017-07-25 13:09:00 -04:00
|
|
|
@wiki_page = described_class.new(wiki, @page, true)
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the slug attribute" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@wiki_page.slug).to eq("test-page")
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the title attribute" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@wiki_page.title).to eq("test page")
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the formatted content attribute" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@wiki_page.content).to eq("test content")
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the format attribute" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@wiki_page.format).to eq(:markdown)
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the message attribute" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@wiki_page.message).to eq("test commit")
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the version attribute" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@wiki_page.version).to be_a Gollum::Git::Commit
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "validations" do
|
|
|
|
before do
|
2015-06-22 14:41:00 -04:00
|
|
|
subject.attributes = { title: 'title', content: 'content' }
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "validates presence of title" do
|
|
|
|
subject.attributes.delete(:title)
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(subject.valid?).to be_falsey
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "validates presence of content" do
|
|
|
|
subject.attributes.delete(:content)
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(subject.valid?).to be_falsey
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
2015-06-22 14:41:00 -04:00
|
|
|
@wiki_attr = { title: "Index", content: "Home Page", format: "markdown" }
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "#create" do
|
|
|
|
after do
|
|
|
|
destroy_page("Index")
|
|
|
|
end
|
|
|
|
|
|
|
|
context "with valid attributes" do
|
|
|
|
it "saves the wiki page" do
|
|
|
|
subject.create(@wiki_attr)
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(wiki.find_page("Index")).not_to be_nil
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "returns true" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(subject.create(@wiki_attr)).to eq(true)
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-03-21 21:04:29 -04:00
|
|
|
describe "dot in the title" do
|
|
|
|
let(:title) { 'Index v1.2.3' }
|
|
|
|
|
|
|
|
before do
|
2015-06-22 14:41:00 -04:00
|
|
|
@wiki_attr = { title: title, content: "Home Page", format: "markdown" }
|
2015-03-21 21:04:29 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "#create" do
|
|
|
|
after do
|
|
|
|
destroy_page(title)
|
|
|
|
end
|
|
|
|
|
|
|
|
context "with valid attributes" do
|
|
|
|
it "saves the wiki page" do
|
|
|
|
subject.create(@wiki_attr)
|
|
|
|
expect(wiki.find_page(title)).not_to be_nil
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns true" do
|
|
|
|
expect(subject.create(@wiki_attr)).to eq(true)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#update" do
|
|
|
|
before do
|
|
|
|
create_page(title, "content")
|
|
|
|
@page = wiki.find_page(title)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "updates the content of the page" do
|
2017-07-23 04:19:10 -04:00
|
|
|
@page.update(content: "new content")
|
2015-03-21 21:04:29 -04:00
|
|
|
@page = wiki.find_page(title)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns true" do
|
2017-07-23 04:19:10 -04:00
|
|
|
expect(@page.update(content: "more content")).to be_truthy
|
2015-03-21 21:04:29 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-03-03 22:43:52 -05:00
|
|
|
describe "#update" do
|
|
|
|
before do
|
|
|
|
create_page("Update", "content")
|
|
|
|
@page = wiki.find_page("Update")
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
2017-07-23 04:19:10 -04:00
|
|
|
destroy_page(@page.title)
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
context "with valid attributes" do
|
|
|
|
it "updates the content of the page" do
|
2017-07-23 04:19:10 -04:00
|
|
|
new_content = "new content"
|
|
|
|
|
|
|
|
@page.update(content: new_content)
|
2013-03-03 22:43:52 -05:00
|
|
|
@page = wiki.find_page("Update")
|
2017-07-23 04:19:10 -04:00
|
|
|
|
|
|
|
expect(@page.content).to eq("new content")
|
|
|
|
end
|
|
|
|
|
|
|
|
it "updates the title of the page" do
|
|
|
|
new_title = "Index v.1.2.4"
|
|
|
|
|
|
|
|
@page.update(title: new_title)
|
|
|
|
@page = wiki.find_page(new_title)
|
|
|
|
|
|
|
|
expect(@page.title).to eq(new_title)
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "returns true" do
|
2017-07-23 04:19:10 -04:00
|
|
|
expect(@page.update(content: "more content")).to be_truthy
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
end
|
2017-03-04 09:03:14 -05:00
|
|
|
|
2017-07-23 08:30:10 -04:00
|
|
|
context 'with same last commit sha' do
|
|
|
|
it 'returns true' do
|
2017-07-23 04:19:10 -04:00
|
|
|
expect(@page.update(content: 'more content', last_commit_sha: @page.last_commit_sha)).to be_truthy
|
2017-03-04 09:03:14 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-07-23 08:30:10 -04:00
|
|
|
context 'with different last commit sha' do
|
|
|
|
it 'raises exception' do
|
2017-07-23 04:19:10 -04:00
|
|
|
expect { @page.update(content: 'more content', last_commit_sha: 'xxx') }.to raise_error(WikiPage::PageChangedError)
|
2017-03-04 09:03:14 -05:00
|
|
|
end
|
|
|
|
end
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "#destroy" do
|
|
|
|
before do
|
|
|
|
create_page("Delete Page", "content")
|
|
|
|
@page = wiki.find_page("Delete Page")
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it "deletes the page" do
|
2013-03-03 22:43:52 -05:00
|
|
|
@page.delete
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(wiki.pages).to be_empty
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it "returns true" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@page.delete).to eq(true)
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#versions" do
|
|
|
|
before do
|
|
|
|
create_page("Update", "content")
|
|
|
|
@page = wiki.find_page("Update")
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
destroy_page("Update")
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns an array of all commits for the page" do
|
2017-07-23 04:19:10 -04:00
|
|
|
3.times { |i| @page.update(content: "content #{i}") }
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@page.versions.count).to eq(4)
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-04-09 01:14:16 -04:00
|
|
|
describe "#title" do
|
|
|
|
before do
|
|
|
|
create_page("Title", "content")
|
|
|
|
@page = wiki.find_page("Title")
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
destroy_page("Title")
|
|
|
|
end
|
|
|
|
|
2016-08-01 11:00:44 -04:00
|
|
|
it "replaces a hyphen to a space" do
|
2014-04-09 01:14:16 -04:00
|
|
|
@page.title = "Import-existing-repositories-into-GitLab"
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(@page.title).to eq("Import existing repositories into GitLab")
|
2014-04-09 01:14:16 -04:00
|
|
|
end
|
2017-08-30 11:16:08 -04:00
|
|
|
|
|
|
|
it 'unescapes html' do
|
|
|
|
@page.title = 'foo & bar'
|
|
|
|
|
|
|
|
expect(@page.title).to eq('foo & bar')
|
|
|
|
end
|
2014-04-09 01:14:16 -04:00
|
|
|
end
|
|
|
|
|
2016-12-18 18:22:20 -05:00
|
|
|
describe '#directory' do
|
|
|
|
context 'when the page is at the root directory' do
|
2016-12-26 23:05:53 -05:00
|
|
|
it 'returns an empty string' do
|
2016-12-18 18:22:20 -05:00
|
|
|
create_page('file', 'content')
|
|
|
|
page = wiki.find_page('file')
|
|
|
|
|
2016-12-26 23:05:53 -05:00
|
|
|
expect(page.directory).to eq('')
|
2016-12-18 18:22:20 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the page is inside an actual directory' do
|
|
|
|
it 'returns the full directory hierarchy' do
|
|
|
|
create_page('dir_1/dir_1_1/file', 'content')
|
|
|
|
page = wiki.find_page('dir_1/dir_1_1/file')
|
|
|
|
|
2016-12-26 23:05:53 -05:00
|
|
|
expect(page.directory).to eq('dir_1/dir_1_1')
|
2016-12-18 18:22:20 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-01-27 08:42:31 -05:00
|
|
|
describe '#historical?' do
|
|
|
|
before do
|
|
|
|
create_page('Update', 'content')
|
|
|
|
@page = wiki.find_page('Update')
|
2017-07-23 04:19:10 -04:00
|
|
|
3.times { |i| @page.update(content: "content #{i}") }
|
2016-01-27 08:42:31 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
destroy_page('Update')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns true when requesting an old version' do
|
|
|
|
old_version = @page.versions.last.to_s
|
|
|
|
old_page = wiki.find_page('Update', old_version)
|
|
|
|
|
|
|
|
expect(old_page.historical?).to eq true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false when requesting latest version' do
|
|
|
|
latest_version = @page.versions.first.to_s
|
|
|
|
latest_page = wiki.find_page('Update', latest_version)
|
|
|
|
|
|
|
|
expect(latest_page.historical?).to eq false
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false when version is nil' do
|
|
|
|
latest_page = wiki.find_page('Update', nil)
|
|
|
|
|
|
|
|
expect(latest_page.historical?).to eq false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-26 20:51:34 -05:00
|
|
|
describe '#to_partial_path' do
|
|
|
|
it 'returns the relative path to the partial to be used' do
|
|
|
|
page = build(:wiki_page)
|
|
|
|
|
|
|
|
expect(page.to_partial_path).to eq('projects/wikis/wiki_page')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-30 08:26:30 -05:00
|
|
|
describe '#==' do
|
|
|
|
let(:original_wiki_page) { create(:wiki_page) }
|
|
|
|
|
|
|
|
it 'returns true for identical wiki page' do
|
|
|
|
expect(original_wiki_page).to eq(original_wiki_page)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false for updated wiki page' do
|
2017-07-23 04:19:10 -04:00
|
|
|
updated_wiki_page = original_wiki_page.update(content: "Updated content")
|
2016-12-30 08:26:30 -05:00
|
|
|
expect(original_wiki_page).not_to eq(updated_wiki_page)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-07-27 09:00:06 -04:00
|
|
|
describe '#last_commit_sha' do
|
|
|
|
before do
|
|
|
|
create_page("Update", "content")
|
|
|
|
@page = wiki.find_page("Update")
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
destroy_page("Update")
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns commit sha' do
|
|
|
|
expect(@page.last_commit_sha).to eq @page.commit.sha
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is changed after page updated' do
|
|
|
|
last_commit_sha_before_update = @page.last_commit_sha
|
|
|
|
|
2017-07-23 04:19:10 -04:00
|
|
|
@page.update(content: "new content")
|
2017-07-27 09:00:06 -04:00
|
|
|
@page = wiki.find_page("Update")
|
|
|
|
|
|
|
|
expect(@page.last_commit_sha).not_to eq last_commit_sha_before_update
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-07-31 12:24:53 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def remove_temp_repo(path)
|
|
|
|
FileUtils.rm_rf path
|
|
|
|
end
|
|
|
|
|
|
|
|
def commit_details
|
2015-10-03 02:48:54 -04:00
|
|
|
{ name: user.name, email: user.email, message: "test commit" }
|
2014-07-31 12:24:53 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def create_page(name, content)
|
|
|
|
wiki.wiki.write_page(name, :markdown, content, commit_details)
|
|
|
|
end
|
|
|
|
|
|
|
|
def destroy_page(title)
|
|
|
|
page = wiki.wiki.paged(title)
|
|
|
|
wiki.wiki.delete_page(page, commit_details)
|
|
|
|
end
|
2016-12-26 17:12:15 -05:00
|
|
|
|
|
|
|
def get_slugs(page_or_dir)
|
|
|
|
if page_or_dir.is_a? WikiPage
|
|
|
|
[page_or_dir.slug]
|
|
|
|
else
|
|
|
|
page_or_dir.pages.present? ? page_or_dir.pages.map(&:slug) : []
|
|
|
|
end
|
|
|
|
end
|
2013-03-03 22:43:52 -05:00
|
|
|
end
|