123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635 |
- /*
- * SonarQube
- * Copyright (C) 2009-2021 SonarSource SA
- * mailto:info AT sonarsource DOT com
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 3 of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public License
- * along with this program; if not, write to the Free Software Foundation,
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
- package org.sonar.server.issue.index;
-
- import java.time.ZoneId;
- import java.util.Date;
- import java.util.Map;
- import java.util.TimeZone;
- import org.elasticsearch.action.search.SearchResponse;
- import org.junit.Rule;
- import org.junit.Test;
- import org.junit.rules.ExpectedException;
- import org.sonar.api.impl.utils.TestSystem2;
- import org.sonar.api.rules.RuleType;
- import org.sonar.api.utils.System2;
- import org.sonar.db.DbTester;
- import org.sonar.db.component.ComponentDto;
- import org.sonar.db.rule.RuleDefinitionDto;
- import org.sonar.server.es.EsTester;
- import org.sonar.server.es.Facets;
- import org.sonar.server.es.SearchOptions;
- import org.sonar.server.permission.index.IndexPermissions;
- import org.sonar.server.permission.index.PermissionIndexerTester;
- import org.sonar.server.permission.index.WebAuthorizationTypeSupport;
- import org.sonar.server.security.SecurityStandards.SQCategory;
- import org.sonar.server.tester.UserSessionRule;
-
- import static java.util.Arrays.asList;
- import static java.util.Arrays.stream;
- import static java.util.Collections.singletonList;
- import static java.util.TimeZone.getTimeZone;
- import static java.util.stream.Collectors.toList;
- import static java.util.stream.IntStream.rangeClosed;
- import static org.assertj.core.api.Assertions.assertThat;
- import static org.assertj.core.api.Assertions.entry;
- import static org.junit.rules.ExpectedException.none;
- import static org.sonar.api.issue.Issue.RESOLUTION_FALSE_POSITIVE;
- import static org.sonar.api.issue.Issue.RESOLUTION_FIXED;
- import static org.sonar.api.issue.Issue.RESOLUTION_REMOVED;
- import static org.sonar.api.issue.Issue.RESOLUTION_WONT_FIX;
- import static org.sonar.api.issue.Issue.STATUS_CLOSED;
- import static org.sonar.api.issue.Issue.STATUS_CONFIRMED;
- import static org.sonar.api.issue.Issue.STATUS_OPEN;
- import static org.sonar.api.issue.Issue.STATUS_REOPENED;
- import static org.sonar.api.issue.Issue.STATUS_RESOLVED;
- import static org.sonar.api.resources.Qualifiers.PROJECT;
- import static org.sonar.api.rule.Severity.BLOCKER;
- import static org.sonar.api.rule.Severity.CRITICAL;
- import static org.sonar.api.rule.Severity.INFO;
- import static org.sonar.api.rule.Severity.MAJOR;
- import static org.sonar.api.rule.Severity.MINOR;
- import static org.sonar.api.utils.DateUtils.parseDateTime;
- import static org.sonar.db.component.ComponentTesting.newDirectory;
- import static org.sonar.db.component.ComponentTesting.newFileDto;
- import static org.sonar.db.component.ComponentTesting.newPrivateProjectDto;
- import static org.sonar.db.rule.RuleTesting.newRule;
- import static org.sonar.server.issue.IssueDocTesting.newDoc;
-
- public class IssueIndexFacetsTest {
-
- @Rule
- public EsTester es = EsTester.create();
- @Rule
- public UserSessionRule userSessionRule = UserSessionRule.standalone();
- @Rule
- public ExpectedException expectedException = none();
- private final TimeZone defaultTimezone = getTimeZone("GMT-01:00");
- private System2 system2 = new TestSystem2().setNow(1_500_000_000_000L).setDefaultTimeZone(defaultTimezone);
- @Rule
- public DbTester db = DbTester.create(system2);
-
- private IssueIndexer issueIndexer = new IssueIndexer(es.client(), db.getDbClient(), new IssueIteratorFactory(db.getDbClient()), null);
- private PermissionIndexerTester authorizationIndexer = new PermissionIndexerTester(es, issueIndexer);
-
- private IssueIndex underTest = new IssueIndex(es.client(), system2, userSessionRule, new WebAuthorizationTypeSupport(userSessionRule));
-
- @Test
- public void facet_on_projectUuids() {
- ComponentDto project = newPrivateProjectDto("ABCD");
- ComponentDto project2 = newPrivateProjectDto("EFGH");
-
- indexIssues(
- newDoc("I1", newFileDto(project, null)),
- newDoc("I2", newFileDto(project, null)),
- newDoc("I3", newFileDto(project2, null)));
-
- assertThatFacetHasExactly(IssueQuery.builder(), "projects", entry("ABCD", 2L), entry("EFGH", 1L));
- }
-
- @Test
- public void facet_on_projectUuids_return_100_entries_plus_selected_values() {
-
- indexIssues(rangeClosed(1, 110).mapToObj(i -> newDoc(newPrivateProjectDto("a" + i))).toArray(IssueDoc[]::new));
- IssueDoc issue1 = newDoc(newPrivateProjectDto("project1"));
- IssueDoc issue2 = newDoc(newPrivateProjectDto("project2"));
- indexIssues(issue1, issue2);
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "projects", 100);
- assertThatFacetHasSize(IssueQuery.builder().projectUuids(asList(issue1.projectUuid(), issue2.projectUuid())).build(), "projects", 102);
- }
-
- @Test
- public void facets_on_files() {
- ComponentDto project = newPrivateProjectDto("A");
- ComponentDto dir = newDirectory(project, "src");
- ComponentDto file1 = newFileDto(project, dir, "ABCD");
- ComponentDto file2 = newFileDto(project, dir, "BCDE");
- ComponentDto file3 = newFileDto(project, dir, "CDEF");
-
- indexIssues(
- newDoc("I1", project),
- newDoc("I2", file1),
- newDoc("I3", file2),
- newDoc("I4", file2),
- newDoc("I5", file3));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "files", entry("src/NAME_ABCD", 1L), entry("src/NAME_BCDE", 2L), entry("src/NAME_CDEF", 1L));
- }
-
- @Test
- public void facet_on_files_return_100_entries_plus_selected_values() {
- ComponentDto project = newPrivateProjectDto();
- indexIssues(rangeClosed(1, 110).mapToObj(i -> newDoc(newFileDto(project, null, "a" + i))).toArray(IssueDoc[]::new));
- IssueDoc issue1 = newDoc(newFileDto(project, null, "file1"));
- IssueDoc issue2 = newDoc(newFileDto(project, null, "file2"));
- indexIssues(issue1, issue2);
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "files", 100);
- assertThatFacetHasSize(IssueQuery.builder().files(asList(issue1.filePath(), issue2.filePath())).build(), "files", 102);
- }
-
- @Test
- public void facets_on_directories() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file1 = newFileDto(project, null).setPath("src/main/xoo/F1.xoo");
- ComponentDto file2 = newFileDto(project, null).setPath("F2.xoo");
-
- indexIssues(
- newDoc("I1", file1).setDirectoryPath("/src/main/xoo"),
- newDoc("I2", file2).setDirectoryPath("/"));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "directories", entry("/src/main/xoo", 1L), entry("/", 1L));
- }
-
- @Test
- public void facet_on_directories_return_100_entries_plus_selected_values() {
- ComponentDto project = newPrivateProjectDto();
- indexIssues(rangeClosed(1, 110).mapToObj(i -> newDoc(newFileDto(project, newDirectory(project, "dir" + i))).setDirectoryPath("a" + i)).toArray(IssueDoc[]::new));
- IssueDoc issue1 = newDoc(newFileDto(project, newDirectory(project, "path1"))).setDirectoryPath("directory1");
- IssueDoc issue2 = newDoc(newFileDto(project, newDirectory(project, "path2"))).setDirectoryPath("directory2");
- indexIssues(issue1, issue2);
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "directories", 100);
- assertThatFacetHasSize(IssueQuery.builder().directories(asList(issue1.directoryPath(), issue2.directoryPath())).build(), "directories", 102);
- }
-
- @Test
- public void facets_on_cwe() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setType(RuleType.VULNERABILITY).setCwe(asList("20", "564", "89", "943")),
- newDoc("I2", file).setType(RuleType.VULNERABILITY).setCwe(asList("943")),
- newDoc("I3", file));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "cwe",
- entry("943", 2L),
- entry("20", 1L),
- entry("564", 1L),
- entry("89", 1L));
- }
-
- @Test
- public void facets_on_owaspTop10() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setType(RuleType.VULNERABILITY).setOwaspTop10(asList("a1", "a2")),
- newDoc("I2", file).setType(RuleType.VULNERABILITY).setOwaspTop10(singletonList("a3")),
- newDoc("I3", file));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "owaspTop10",
- entry("a1", 1L),
- entry("a2", 1L),
- entry("a3", 1L));
- }
-
- @Test
- public void facets_on_sansTop25() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setType(RuleType.VULNERABILITY).setSansTop25(asList("porous-defenses", "risky-resource", "insecure-interaction")),
- newDoc("I2", file).setType(RuleType.VULNERABILITY).setSansTop25(singletonList("porous-defenses")),
- newDoc("I3", file));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "sansTop25",
- entry("insecure-interaction", 1L),
- entry("porous-defenses", 2L),
- entry("risky-resource", 1L));
- }
-
- @Test
- public void facets_on_sonarSourceSecurity() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setType(RuleType.VULNERABILITY).setSonarSourceSecurityCategory(SQCategory.BUFFER_OVERFLOW),
- newDoc("I2", file).setType(RuleType.VULNERABILITY).setSonarSourceSecurityCategory(SQCategory.DOS),
- newDoc("I3", file));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "sonarsourceSecurity",
- entry("buffer-overflow", 1L),
- entry("dos", 1L));
- }
-
- @Test
- public void facets_on_severities() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setSeverity(INFO),
- newDoc("I2", file).setSeverity(INFO),
- newDoc("I3", file).setSeverity(MAJOR));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "severities", entry("INFO", 2L), entry("MAJOR", 1L));
- }
-
- @Test
- public void facet_on_severities_return_5_entries_max() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I2", file).setSeverity(INFO),
- newDoc("I1", file).setSeverity(MINOR),
- newDoc("I3", file).setSeverity(MAJOR),
- newDoc("I4", file).setSeverity(CRITICAL),
- newDoc("I5", file).setSeverity(BLOCKER),
- newDoc("I6", file).setSeverity(MAJOR));
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "severities", 5);
- }
-
- @Test
- public void facets_on_statuses() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setStatus(STATUS_CLOSED),
- newDoc("I2", file).setStatus(STATUS_CLOSED),
- newDoc("I3", file).setStatus(STATUS_OPEN));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "statuses", entry("CLOSED", 2L), entry("OPEN", 1L));
- }
-
- @Test
- public void facet_on_statuses_return_5_entries_max() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setStatus(STATUS_OPEN),
- newDoc("I2", file).setStatus(STATUS_CONFIRMED),
- newDoc("I3", file).setStatus(STATUS_REOPENED),
- newDoc("I4", file).setStatus(STATUS_RESOLVED),
- newDoc("I5", file).setStatus(STATUS_CLOSED),
- newDoc("I6", file).setStatus(STATUS_OPEN));
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "statuses", 5);
- }
-
- @Test
- public void facets_on_resolutions() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setResolution(RESOLUTION_FALSE_POSITIVE),
- newDoc("I2", file).setResolution(RESOLUTION_FALSE_POSITIVE),
- newDoc("I3", file).setResolution(RESOLUTION_FIXED));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "resolutions", entry("FALSE-POSITIVE", 2L), entry("FIXED", 1L));
- }
-
- @Test
- public void facets_on_resolutions_return_5_entries_max() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setResolution(RESOLUTION_FIXED),
- newDoc("I2", file).setResolution(RESOLUTION_FALSE_POSITIVE),
- newDoc("I3", file).setResolution(RESOLUTION_REMOVED),
- newDoc("I4", file).setResolution(RESOLUTION_WONT_FIX),
- newDoc("I5", file).setResolution(null));
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "resolutions", 5);
- }
-
- @Test
- public void facets_on_languages() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
- RuleDefinitionDto ruleDefinitionDto = newRule();
- db.rules().insert(ruleDefinitionDto);
-
- indexIssues(newDoc("I1", file).setRuleUuid(ruleDefinitionDto.getUuid()).setLanguage("xoo"));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "languages", entry("xoo", 1L));
- }
-
- @Test
- public void facets_on_languages_return_100_entries_plus_selected_values() {
- ComponentDto project = newPrivateProjectDto();
- indexIssues(rangeClosed(1, 100).mapToObj(i -> newDoc(newFileDto(project, null)).setLanguage("a" + i)).toArray(IssueDoc[]::new));
- IssueDoc issue1 = newDoc(newFileDto(project, null)).setLanguage("language1");
- IssueDoc issue2 = newDoc(newFileDto(project, null)).setLanguage("language2");
- indexIssues(issue1, issue2);
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "languages", 100);
- assertThatFacetHasSize(IssueQuery.builder().languages(asList(issue1.language(), issue2.language())).build(), "languages", 102);
- }
-
- @Test
- public void facets_on_assignees() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setAssigneeUuid("steph-uuid"),
- newDoc("I2", file).setAssigneeUuid("marcel-uuid"),
- newDoc("I3", file).setAssigneeUuid("marcel-uuid"),
- newDoc("I4", file).setAssigneeUuid(null));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "assignees", entry("steph-uuid", 1L), entry("marcel-uuid", 2L), entry("", 1L));
- }
-
- @Test
- public void facets_on_assignees_return_only_100_entries_plus_selected_values() {
- ComponentDto project = newPrivateProjectDto();
- indexIssues(rangeClosed(1, 110).mapToObj(i -> newDoc(newFileDto(project, null)).setAssigneeUuid("a" + i)).toArray(IssueDoc[]::new));
- IssueDoc issue1 = newDoc(newFileDto(project, null)).setAssigneeUuid("user1");
- IssueDoc issue2 = newDoc(newFileDto(project, null)).setAssigneeUuid("user2");
- indexIssues(issue1, issue2);
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "assignees", 100);
- assertThatFacetHasSize(IssueQuery.builder().assigneeUuids(asList(issue1.assigneeUuid(), issue2.assigneeUuid())).build(), "assignees", 102);
- }
-
- @Test
- public void facets_on_assignees_supports_dashes() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setAssigneeUuid("j-b-uuid"),
- newDoc("I2", file).setAssigneeUuid("marcel-uuid"),
- newDoc("I3", file).setAssigneeUuid("marcel-uuid"),
- newDoc("I4", file).setAssigneeUuid(null));
-
- assertThatFacetHasOnly(IssueQuery.builder().assigneeUuids(singletonList("j-b")),
- "assignees", entry("j-b-uuid", 1L), entry("marcel-uuid", 2L), entry("", 1L));
- }
-
- @Test
- public void facets_on_author() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- indexIssues(
- newDoc("I1", file).setAuthorLogin("steph"),
- newDoc("I2", file).setAuthorLogin("marcel"),
- newDoc("I3", file).setAuthorLogin("marcel"),
- newDoc("I4", file).setAuthorLogin(null));
-
- assertThatFacetHasOnly(IssueQuery.builder(), "author", entry("steph", 1L), entry("marcel", 2L));
- }
-
- @Test
- public void facets_on_authors_return_100_entries_plus_selected_values() {
- ComponentDto project = newPrivateProjectDto();
- indexIssues(rangeClosed(1, 110).mapToObj(i -> newDoc(newFileDto(project, null)).setAuthorLogin("a" + i)).toArray(IssueDoc[]::new));
- IssueDoc issue1 = newDoc(newFileDto(project, null)).setAuthorLogin("user1");
- IssueDoc issue2 = newDoc(newFileDto(project, null)).setAuthorLogin("user2");
- indexIssues(issue1, issue2);
-
- assertThatFacetHasSize(IssueQuery.builder().build(), "author", 100);
- assertThatFacetHasSize(IssueQuery.builder().authors(asList(issue1.authorLogin(), issue2.authorLogin())).build(), "author", 102);
- }
-
- @Test
- public void facet_on_created_at_with_less_than_20_days_use_system_timezone_by_default() {
- SearchOptions options = fixtureForCreatedAtFacet();
-
- IssueQuery query = IssueQuery.builder()
- .createdAfter(parseDateTime("2014-09-01T00:00:00+0100"))
- .createdBefore(parseDateTime("2014-09-08T00:00:00+0100"))
- .build();
- SearchResponse result = underTest.search(query, options);
- Map<String, Long> buckets = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(buckets).containsOnly(
- entry("2014-08-31", 0L),
- entry("2014-09-01", 2L),
- entry("2014-09-02", 1L),
- entry("2014-09-03", 0L),
- entry("2014-09-04", 0L),
- entry("2014-09-05", 1L),
- entry("2014-09-06", 0L),
- entry("2014-09-07", 0L));
- }
-
- @Test
- public void facet_on_created_at_with_less_than_20_days_use_user_timezone_if_provided() {
- // Use timezones very far from each other in order to see some issues moving to a different calendar day
- final ZoneId plus14 = ZoneId.of("Pacific/Kiritimati");
- final ZoneId minus11 = ZoneId.of("Pacific/Pago_Pago");
-
- SearchOptions options = fixtureForCreatedAtFacet();
-
- final Date startDate = parseDateTime("2014-09-01T00:00:00+0000");
- final Date endDate = parseDateTime("2014-09-08T00:00:00+0000");
-
- IssueQuery queryPlus14 = IssueQuery.builder()
- .createdAfter(startDate)
- .createdBefore(endDate)
- .timeZone(plus14)
- .build();
- SearchResponse resultPlus14 = underTest.search(queryPlus14, options);
- Map<String, Long> bucketsPlus14 = new Facets(resultPlus14, plus14).get("createdAt");
- assertThat(bucketsPlus14).containsOnly(
- entry("2014-09-01", 0L),
- entry("2014-09-02", 2L),
- entry("2014-09-03", 1L),
- entry("2014-09-04", 0L),
- entry("2014-09-05", 0L),
- entry("2014-09-06", 1L),
- entry("2014-09-07", 0L),
- entry("2014-09-08", 0L));
-
- IssueQuery queryMinus11 = IssueQuery.builder()
- .createdAfter(startDate)
- .createdBefore(endDate)
- .timeZone(minus11)
- .build();
- SearchResponse resultMinus11 = underTest.search(queryMinus11, options);
- Map<String, Long> bucketsMinus11 = new Facets(resultMinus11, minus11).get("createdAt");
- assertThat(bucketsMinus11).containsOnly(
- entry("2014-08-31", 1L),
- entry("2014-09-01", 1L),
- entry("2014-09-02", 1L),
- entry("2014-09-03", 0L),
- entry("2014-09-04", 0L),
- entry("2014-09-05", 1L),
- entry("2014-09-06", 0L),
- entry("2014-09-07", 0L));
- }
-
- @Test
- public void facet_on_created_at_with_less_than_20_weeks() {
- SearchOptions options = fixtureForCreatedAtFacet();
-
- SearchResponse result = underTest.search(IssueQuery.builder()
- .createdAfter(parseDateTime("2014-09-01T00:00:00+0100"))
- .createdBefore(parseDateTime("2014-09-21T00:00:00+0100")).build(),
- options);
- Map<String, Long> createdAt = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(createdAt).containsOnly(
- entry("2014-08-25", 0L),
- entry("2014-09-01", 4L),
- entry("2014-09-08", 0L),
- entry("2014-09-15", 1L));
- }
-
- @Test
- public void facet_on_created_at_with_less_than_20_months() {
- SearchOptions options = fixtureForCreatedAtFacet();
-
- SearchResponse result = underTest.search(IssueQuery.builder()
- .createdAfter(parseDateTime("2014-09-01T00:00:00+0100"))
- .createdBefore(parseDateTime("2015-01-19T00:00:00+0100")).build(),
- options);
- Map<String, Long> createdAt = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(createdAt).containsOnly(
- entry("2014-08-01", 0L),
- entry("2014-09-01", 5L),
- entry("2014-10-01", 0L),
- entry("2014-11-01", 0L),
- entry("2014-12-01", 0L),
- entry("2015-01-01", 1L));
- }
-
- @Test
- public void facet_on_created_at_with_more_than_20_months() {
- SearchOptions options = fixtureForCreatedAtFacet();
-
- SearchResponse result = underTest.search(IssueQuery.builder()
- .createdAfter(parseDateTime("2011-01-01T00:00:00+0100"))
- .createdBefore(parseDateTime("2016-01-01T00:00:00+0100")).build(),
- options);
- Map<String, Long> createdAt = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(createdAt).containsOnly(
- entry("2010-01-01", 0L),
- entry("2011-01-01", 1L),
- entry("2012-01-01", 0L),
- entry("2013-01-01", 0L),
- entry("2014-01-01", 5L),
- entry("2015-01-01", 1L));
- }
-
- @Test
- public void facet_on_created_at_with_one_day() {
- SearchOptions options = fixtureForCreatedAtFacet();
-
- SearchResponse result = underTest.search(IssueQuery.builder()
- .createdAfter(parseDateTime("2014-09-01T00:00:00-0100"))
- .createdBefore(parseDateTime("2014-09-02T00:00:00-0100")).build(),
- options);
- Map<String, Long> createdAt = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(createdAt).containsOnly(
- entry("2014-09-01", 2L));
- }
-
- @Test
- public void facet_on_created_at_with_bounds_outside_of_data() {
- SearchOptions options = fixtureForCreatedAtFacet();
-
- SearchResponse result = underTest.search(IssueQuery.builder()
- .createdAfter(parseDateTime("2009-01-01T00:00:00+0100"))
- .createdBefore(parseDateTime("2016-01-01T00:00:00+0100"))
- .build(), options);
- Map<String, Long> createdAt = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(createdAt).containsOnly(
- entry("2008-01-01", 0L),
- entry("2009-01-01", 0L),
- entry("2010-01-01", 0L),
- entry("2011-01-01", 1L),
- entry("2012-01-01", 0L),
- entry("2013-01-01", 0L),
- entry("2014-01-01", 5L),
- entry("2015-01-01", 1L));
- }
-
- @Test
- public void facet_on_created_at_without_start_bound() {
- SearchOptions searchOptions = fixtureForCreatedAtFacet();
-
- SearchResponse result = underTest.search(IssueQuery.builder()
- .createdBefore(parseDateTime("2016-01-01T00:00:00+0100")).build(),
- searchOptions);
- Map<String, Long> createdAt = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(createdAt).containsOnly(
- entry("2011-01-01", 1L),
- entry("2012-01-01", 0L),
- entry("2013-01-01", 0L),
- entry("2014-01-01", 5L),
- entry("2015-01-01", 1L));
- }
-
- @Test
- public void facet_on_created_at_without_issues() {
- SearchOptions searchOptions = new SearchOptions().addFacets("createdAt");
-
- SearchResponse result = underTest.search(IssueQuery.builder().build(), searchOptions);
- Map<String, Long> createdAt = new Facets(result, system2.getDefaultTimeZone().toZoneId()).get("createdAt");
- assertThat(createdAt).isNull();
- }
-
- private SearchOptions fixtureForCreatedAtFacet() {
- ComponentDto project = newPrivateProjectDto();
- ComponentDto file = newFileDto(project, null);
-
- IssueDoc issue0 = newDoc("ISSUE0", file).setFuncCreationDate(parseDateTime("2011-04-25T00:05:13+0000"));
- IssueDoc issue1 = newDoc("I1", file).setFuncCreationDate(parseDateTime("2014-09-01T10:34:56+0000"));
- IssueDoc issue2 = newDoc("I2", file).setFuncCreationDate(parseDateTime("2014-09-01T22:46:00+0000"));
- IssueDoc issue3 = newDoc("I3", file).setFuncCreationDate(parseDateTime("2014-09-02T11:34:56+0000"));
- IssueDoc issue4 = newDoc("I4", file).setFuncCreationDate(parseDateTime("2014-09-05T11:34:56+0000"));
- IssueDoc issue5 = newDoc("I5", file).setFuncCreationDate(parseDateTime("2014-09-20T11:34:56+0000"));
- IssueDoc issue6 = newDoc("I6", file).setFuncCreationDate(parseDateTime("2015-01-18T11:34:56+0000"));
-
- indexIssues(issue0, issue1, issue2, issue3, issue4, issue5, issue6);
-
- return new SearchOptions().addFacets("createdAt");
- }
-
- private void indexIssues(IssueDoc... issues) {
- issueIndexer.index(asList(issues).iterator());
- authorizationIndexer.allow(stream(issues).map(issue -> new IndexPermissions(issue.projectUuid(), PROJECT).allowAnyone()).collect(toList()));
- }
-
- @SafeVarargs
- private final void assertThatFacetHasExactly(IssueQuery.Builder query, String facet, Map.Entry<String, Long>... expectedEntries) {
- SearchResponse result = underTest.search(query.build(), new SearchOptions().addFacets(singletonList(facet)));
- Facets facets = new Facets(result, system2.getDefaultTimeZone().toZoneId());
- assertThat(facets.getNames()).containsOnly(facet, "effort");
- assertThat(facets.get(facet)).containsExactly(expectedEntries);
- }
-
- @SafeVarargs
- private final void assertThatFacetHasOnly(IssueQuery.Builder query, String facet, Map.Entry<String, Long>... expectedEntries) {
- SearchResponse result = underTest.search(query.build(), new SearchOptions().addFacets(singletonList(facet)));
- Facets facets = new Facets(result, system2.getDefaultTimeZone().toZoneId());
- assertThat(facets.getNames()).containsOnly(facet, "effort");
- assertThat(facets.get(facet)).containsOnly(expectedEntries);
- }
-
- private void assertThatFacetHasSize(IssueQuery issueQuery, String facet, int expectedSize) {
- SearchResponse result = underTest.search(issueQuery, new SearchOptions().addFacets(singletonList(facet)));
- Facets facets = new Facets(result, system2.getDefaultTimeZone().toZoneId());
- assertThat(facets.get(facet)).hasSize(expectedSize);
- }
- }
|