mirror of https://github.com/nocodb/nocodb
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
509 lines
15 KiB
509 lines
15 KiB
2 years ago
|
import { mainPage } from "../../support/page_objects/mainPage";
|
||
|
import { isTestSuiteActive } from "../../support/page_objects/projectConstants";
|
||
2 years ago
|
import { loginPage } from "../../support/page_objects/navigation";
|
||
2 years ago
|
|
||
|
// kanban grouping field configuration
|
||
|
//
|
||
|
function configureGroupingField(field, closeMenu = true) {
|
||
|
cy.get(".nc-kanban-stacked-by-menu-btn").click();
|
||
|
|
||
2 years ago
|
cy.getActiveMenu(".nc-dropdown-kanban-stacked-by-menu")
|
||
|
.should("exist")
|
||
|
.find(".nc-kanban-grouping-field-select")
|
||
2 years ago
|
.click();
|
||
2 years ago
|
cy.get(".ant-select-dropdown:visible")
|
||
|
.should("exist")
|
||
2 years ago
|
.find(`.ant-select-item`)
|
||
|
.contains(new RegExp("^" + field + "$", "g"))
|
||
2 years ago
|
.should("exist")
|
||
2 years ago
|
.click();
|
||
|
|
||
|
if (closeMenu) {
|
||
2 years ago
|
cy.get(".nc-kanban-stacked-by-menu-btn").click();
|
||
2 years ago
|
}
|
||
|
|
||
2 years ago
|
cy.get(".nc-kanban-stacked-by-menu-btn")
|
||
2 years ago
|
.contains(`Stacked By ${field}`)
|
||
2 years ago
|
.should("exist");
|
||
2 years ago
|
}
|
||
|
|
||
|
// number of kanban stacks altogether
|
||
|
//
|
||
|
function verifyKanbanStackCount(count) {
|
||
2 years ago
|
cy.get(".nc-kanban-stack").should("have.length", count);
|
||
2 years ago
|
}
|
||
|
|
||
|
// order of kanban stacks
|
||
|
//
|
||
|
function verifyKanbanStackOrder(order) {
|
||
2 years ago
|
cy.get(".nc-kanban-stack").each(($el, index) => {
|
||
|
cy.wrap($el).should("contain", order[index]);
|
||
2 years ago
|
});
|
||
|
}
|
||
|
|
||
|
// kanban stack footer numbers
|
||
|
//
|
||
|
function verifyKanbanStackFooterCount(count) {
|
||
2 years ago
|
cy.get(".nc-kanban-stack").each(($el, index) => {
|
||
|
cy.wrap($el)
|
||
|
.find(".nc-kanban-data-count")
|
||
2 years ago
|
.should(
|
||
|
"contain",
|
||
|
`${count[index]} record${count[index] > 1 ? "s" : ""}`
|
||
|
);
|
||
2 years ago
|
});
|
||
|
}
|
||
|
|
||
|
// kanban card count in a stack
|
||
|
//
|
||
|
function verifyKanbanStackCardCount(count) {
|
||
2 years ago
|
cy.get(".nc-kanban-stack").each(($el, index) => {
|
||
|
if (count[index] > 0) {
|
||
|
cy.wrap($el)
|
||
|
.find(".nc-kanban-item")
|
||
|
.should("exist")
|
||
|
.should("have.length", count[index]);
|
||
2 years ago
|
}
|
||
|
});
|
||
|
}
|
||
|
|
||
|
// order of cards within a stack
|
||
|
//
|
||
|
function verifyKanbanStackCardOrder(order, stackIndex, cardIndex) {
|
||
2 years ago
|
cy.get(".nc-kanban-stack")
|
||
|
.eq(stackIndex)
|
||
|
.find(".nc-kanban-item")
|
||
|
.eq(cardIndex)
|
||
|
.should("contain", order);
|
||
2 years ago
|
}
|
||
|
|
||
|
// drag drop kanban card
|
||
|
//
|
||
|
function dragAndDropKanbanCard(srcCard, dstCard) {
|
||
2 years ago
|
cy.get(`.nc-kanban-item .ant-card :visible:contains("${srcCard}")`).drag(
|
||
|
`.nc-kanban-item :visible:contains("${dstCard}")`
|
||
|
);
|
||
2 years ago
|
}
|
||
|
|
||
|
// drag drop kanban stack
|
||
|
//
|
||
|
function dragAndDropKanbanStack(srcStack, dstStack) {
|
||
2 years ago
|
cy.get(`.nc-kanban-stack-head :contains("${srcStack}")`).drag(
|
||
|
`.nc-kanban-stack-head :contains("${dstStack}")`
|
||
|
);
|
||
2 years ago
|
}
|
||
|
|
||
2 years ago
|
let localDebug = false;
|
||
|
|
||
2 years ago
|
// test suite
|
||
|
//
|
||
|
export const genTest = (apiType, dbType) => {
|
||
|
if (!isTestSuiteActive(apiType, dbType)) return;
|
||
|
|
||
|
describe(`${apiType.toUpperCase()} api - Kanban`, () => {
|
||
|
before(() => {
|
||
2 years ago
|
// if (localDebug) {
|
||
|
// // for standalone tests
|
||
|
// cy.restoreLocalStorage();
|
||
|
// loginPage.loginAndOpenProject(apiType, dbType);
|
||
|
//
|
||
|
// cy.openTableTab("Film", 25);
|
||
|
// cy.openTableView("kanban", "Kanban-1");
|
||
|
//
|
||
|
// cy.saveLocalStorageToFile("kanban");
|
||
|
// }
|
||
|
// cy.restoreLocalStorageFromFile("kanban");
|
||
|
// cy.wait(1000);
|
||
|
// cy.visit(
|
||
|
// "http://localhost:3000/#/nc/p_42i93khqhge32z/table/Film/Kanban-1",
|
||
|
// { baseUrl: null }
|
||
|
// );
|
||
|
// verifyKanbanStackCount(7);
|
||
|
// verifyKanbanStackOrder([
|
||
|
// "uncategorized",
|
||
|
// "G",
|
||
|
// "PG",
|
||
|
// "PG-13",
|
||
|
// "R",
|
||
|
// "NC-17",
|
||
|
// "Test",
|
||
|
// ]);
|
||
2 years ago
|
});
|
||
|
|
||
|
beforeEach(() => {
|
||
|
cy.restoreLocalStorage();
|
||
|
});
|
||
|
|
||
|
afterEach(() => {
|
||
|
cy.saveLocalStorage();
|
||
|
});
|
||
|
|
||
2 years ago
|
after(() => {});
|
||
2 years ago
|
|
||
2 years ago
|
/**
|
||
2 years ago
|
class name specific to kanban view
|
||
2 years ago
|
.nc-kanban-stacked-by-menu-btn
|
||
|
.nc-dropdown-kanban-stacked-by-menu
|
||
|
.nc-kanban-add-edit-stack-menu-btn
|
||
|
.nc-dropdown-kanban-add-edit-stack-menu
|
||
|
.nc-kanban-grouping-field-select
|
||
|
.nc-dropdown-kanban-stack-context-menu
|
||
|
**/
|
||
|
|
||
2 years ago
|
it("Create Kanban view", () => {
|
||
2 years ago
|
if (localDebug === false) {
|
||
2 years ago
|
cy.openTableTab("Film", 25);
|
||
|
cy.viewCreate("kanban");
|
||
|
}
|
||
2 years ago
|
});
|
||
|
|
||
|
it("Rename Kanban view", () => {
|
||
|
cy.viewRename("kanban", 0, "Film Kanban");
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
|
it("Configure grouping field", () => {
|
||
2 years ago
|
configureGroupingField("Rating", true);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
|
it("Verify kanban stacks", () => {
|
||
|
verifyKanbanStackCount(6);
|
||
2 years ago
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"PG-13",
|
||
|
"R",
|
||
|
"NC-17",
|
||
|
]);
|
||
2 years ago
|
verifyKanbanStackFooterCount(["0", "178", "194", "223", "195", "210"]);
|
||
|
verifyKanbanStackCardCount([0, 25, 25, 25, 25, 25]);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
|
it("Hide fields", () => {
|
||
|
mainPage.hideAllColumns();
|
||
2 years ago
|
mainPage.unhideField("Title", "kanban");
|
||
2 years ago
|
|
||
|
verifyKanbanStackCardCount([0, 25, 25, 25, 25, 25]);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
|
it("Verify card order", () => {
|
||
|
// verify 3 cards from each stack
|
||
|
verifyKanbanStackCardOrder("ACE GOLDFINGER", 1, 0);
|
||
|
verifyKanbanStackCardOrder("AFFAIR PREJUDICE", 1, 1);
|
||
|
verifyKanbanStackCardOrder("AFRICAN EGG", 1, 2);
|
||
|
|
||
|
verifyKanbanStackCardOrder("ACADEMY DINOSAUR", 2, 0);
|
||
|
verifyKanbanStackCardOrder("AGENT TRUMAN", 2, 1);
|
||
|
verifyKanbanStackCardOrder("ALASKA PHANTOM", 2, 2);
|
||
|
|
||
|
verifyKanbanStackCardOrder("AIRPLANE SIERRA", 3, 0);
|
||
|
verifyKanbanStackCardOrder("ALABAMA DEVIL", 3, 1);
|
||
|
verifyKanbanStackCardOrder("ALTER VICTORY", 3, 2);
|
||
|
|
||
|
verifyKanbanStackCardOrder("AIRPORT POLLOCK", 4, 0);
|
||
|
verifyKanbanStackCardOrder("ALONE TRIP", 4, 1);
|
||
|
verifyKanbanStackCardOrder("AMELIE HELLFIGHTERS", 4, 2);
|
||
|
|
||
|
verifyKanbanStackCardOrder("ADAPTATION HOLES", 5, 0);
|
||
|
verifyKanbanStackCardOrder("ALADDIN CALENDAR", 5, 1);
|
||
|
verifyKanbanStackCardOrder("ALICE FANTASIA", 5, 2);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
2 years ago
|
it.skip("Verify inter-stack drag and drop", () => {
|
||
2 years ago
|
dragAndDropKanbanCard("ACE GOLDFINGER", "ACADEMY DINOSAUR");
|
||
|
verifyKanbanStackCardOrder("AFFAIR PREJUDICE", 1, 0);
|
||
|
verifyKanbanStackCardOrder("ACE GOLDFINGER", 2, 0);
|
||
|
verifyKanbanStackCardOrder("ACADEMY DINOSAUR", 2, 1);
|
||
|
|
||
|
dragAndDropKanbanCard("ACE GOLDFINGER", "AFFAIR PREJUDICE");
|
||
|
verifyKanbanStackCardOrder("ACE GOLDFINGER", 1, 0);
|
||
|
verifyKanbanStackCardOrder("AFFAIR PREJUDICE", 1, 1);
|
||
|
verifyKanbanStackCardOrder("ACADEMY DINOSAUR", 2, 0);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
2 years ago
|
it.skip("Verify intra-stack drag and drop", () => {
|
||
2 years ago
|
dragAndDropKanbanCard("ACE GOLDFINGER", "AFFAIR PREJUDICE");
|
||
|
verifyKanbanStackCardOrder("AFFAIR PREJUDICE", 1, 0);
|
||
|
verifyKanbanStackCardOrder("ACE GOLDFINGER", 1, 1);
|
||
|
|
||
|
dragAndDropKanbanCard("ACE GOLDFINGER", "AFFAIR PREJUDICE");
|
||
|
verifyKanbanStackCardOrder("ACE GOLDFINGER", 1, 0);
|
||
|
verifyKanbanStackCardOrder("AFFAIR PREJUDICE", 1, 1);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
|
it("Verify stack drag drop", () => {
|
||
2 years ago
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"PG-13",
|
||
|
"R",
|
||
|
"NC-17",
|
||
|
]);
|
||
2 years ago
|
dragAndDropKanbanStack("PG-13", "R");
|
||
2 years ago
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"R",
|
||
|
"PG-13",
|
||
|
"NC-17",
|
||
|
]);
|
||
2 years ago
|
dragAndDropKanbanStack("PG-13", "R");
|
||
2 years ago
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"PG-13",
|
||
|
"R",
|
||
|
"NC-17",
|
||
|
]);
|
||
|
});
|
||
2 years ago
|
|
||
|
it("Verify Sort", () => {
|
||
|
mainPage.sortField("Title", "Z → A");
|
||
|
verifyKanbanStackCardOrder("YOUNG LANGUAGE", 1, 0);
|
||
|
verifyKanbanStackCardOrder("WEST LION", 1, 1);
|
||
|
verifyKanbanStackCardOrder("WORST BANGER", 2, 0);
|
||
|
verifyKanbanStackCardOrder("WORDS HUNTER", 2, 1);
|
||
|
|
||
|
mainPage.clearSort();
|
||
|
verifyKanbanStackCardOrder("ACE GOLDFINGER", 1, 0);
|
||
|
verifyKanbanStackCardOrder("AFFAIR PREJUDICE", 1, 1);
|
||
|
verifyKanbanStackCardOrder("ACADEMY DINOSAUR", 2, 0);
|
||
|
verifyKanbanStackCardOrder("AGENT TRUMAN", 2, 1);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
|
it("Verify Filter", () => {
|
||
|
mainPage.filterField("Title", "is like", "BA");
|
||
|
verifyKanbanStackCardOrder("BAKED CLEOPATRA", 1, 0);
|
||
|
verifyKanbanStackCardOrder("BALLROOM MOCKINGBIRD", 1, 1);
|
||
|
verifyKanbanStackCardOrder("ARIZONA BANG", 2, 0);
|
||
|
verifyKanbanStackCardOrder("EGYPT TENENBAUMS", 2, 1);
|
||
|
|
||
|
mainPage.filterReset();
|
||
|
verifyKanbanStackCardOrder("ACE GOLDFINGER", 1, 0);
|
||
|
verifyKanbanStackCardOrder("AFFAIR PREJUDICE", 1, 1);
|
||
|
verifyKanbanStackCardOrder("ACADEMY DINOSAUR", 2, 0);
|
||
|
verifyKanbanStackCardOrder("AGENT TRUMAN", 2, 1);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
2 years ago
|
// it("Stack context menu- rename stack", () => {
|
||
|
// verifyKanbanStackCount(6);
|
||
|
// cy.get('.nc-kanban-stack-head').eq(1).find('.ant-dropdown-trigger').click();
|
||
|
// cy.getActiveMenu('.nc-dropdown-kanban-stack-context-menu').should('be.visible');
|
||
|
// cy.getActiveMenu('.nc-dropdown-kanban-stack-context-menu')
|
||
|
// .find('.ant-dropdown-menu-item')
|
||
|
// .contains('Rename Stack')
|
||
|
// .click();
|
||
|
// })
|
||
|
|
||
2 years ago
|
it("Stack context menu- delete stack", () => {});
|
||
2 years ago
|
|
||
2 years ago
|
it("Stack context menu- collapse stack", () => {});
|
||
2 years ago
|
|
||
|
it("Copy view", () => {
|
||
|
mainPage.sortField("Title", "Z → A");
|
||
|
mainPage.filterField("Title", "is like", "BA");
|
||
|
|
||
|
cy.viewCopy(1);
|
||
|
|
||
|
// verify copied view
|
||
2 years ago
|
cy.get(".nc-kanban-stacked-by-menu-btn")
|
||
2 years ago
|
.contains(`Stacked By Rating`)
|
||
2 years ago
|
.should("exist");
|
||
2 years ago
|
verifyKanbanStackCount(6);
|
||
2 years ago
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"PG-13",
|
||
|
"R",
|
||
|
"NC-17",
|
||
|
]);
|
||
2 years ago
|
verifyKanbanStackFooterCount(["0", "4", "5", "8", "6", "6"]);
|
||
|
verifyKanbanStackCardOrder("BAREFOOT MANCHURIAN", 1, 0);
|
||
|
verifyKanbanStackCardOrder("WORST BANGER", 2, 0);
|
||
|
|
||
|
cy.viewDelete(1);
|
||
2 years ago
|
});
|
||
2 years ago
|
|
||
2 years ago
|
it("Add stack", () => {
|
||
|
cy.get(".nc-kanban-add-edit-stack-menu-btn").should("exist").click();
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-add-edit-stack-menu").should(
|
||
|
"be.visible"
|
||
|
);
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-add-edit-stack-menu")
|
||
|
.find(".ant-btn-dashed")
|
||
|
.click();
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-add-edit-stack-menu")
|
||
|
.find(".nc-select-option")
|
||
|
.last()
|
||
|
.click()
|
||
|
.type("Test{enter}");
|
||
|
verifyKanbanStackCount(7);
|
||
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"PG-13",
|
||
|
"R",
|
||
|
"NC-17",
|
||
|
"Test",
|
||
|
]);
|
||
|
});
|
||
|
|
||
|
it("Collapse stack", () => {
|
||
|
cy.get(".nc-kanban-stack-head").last().scrollIntoView();
|
||
|
cy.get(".nc-kanban-stack-head").last().click();
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-stack-context-menu").should(
|
||
|
"be.visible"
|
||
|
);
|
||
|
|
||
|
// collapse stack
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-stack-context-menu")
|
||
|
.find(".ant-dropdown-menu-item")
|
||
|
.contains("Collapse Stack")
|
||
|
.click();
|
||
|
cy.get(".nc-kanban-collapsed-stack")
|
||
|
.should("exist")
|
||
|
.should("have.length", 1);
|
||
|
|
||
|
// expand back
|
||
|
cy.get(".nc-kanban-collapsed-stack").click();
|
||
|
cy.get(".nc-kanban-collapsed-stack")
|
||
|
.should("not.exist")
|
||
|
.should("have.length", 0);
|
||
|
});
|
||
|
|
||
|
it("Add record to stack", () => {
|
||
|
mainPage.hideAllColumns();
|
||
|
mainPage.toggleShowSystemFields();
|
||
|
mainPage.unhideField("LanguageId", "kanban");
|
||
|
mainPage.unhideField("Title", "kanban");
|
||
|
|
||
|
cy.get(".nc-kanban-stack-head").last().scrollIntoView();
|
||
|
cy.get(".nc-kanban-stack-head").last().click();
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-stack-context-menu").should(
|
||
|
"be.visible"
|
||
|
);
|
||
|
|
||
|
// add record
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-stack-context-menu")
|
||
|
.find(".ant-dropdown-menu-item")
|
||
|
.contains("Add new record")
|
||
|
.click();
|
||
|
|
||
|
cy.getActiveDrawer(".nc-drawer-expanded-form").should("be.visible");
|
||
|
cy.get(".nc-expand-col-Title")
|
||
|
.find(".nc-cell > input")
|
||
|
.should("exist")
|
||
|
.first()
|
||
|
.clear()
|
||
|
.type("New record");
|
||
|
cy.get(".nc-expand-col-LanguageId")
|
||
|
.find(".nc-cell > input")
|
||
|
.should("exist")
|
||
|
.first()
|
||
|
.clear()
|
||
|
.type("1");
|
||
|
|
||
|
cy.getActiveDrawer(".nc-drawer-expanded-form")
|
||
|
.find("button")
|
||
|
.contains("Save row")
|
||
|
.click();
|
||
|
cy.toastWait("updated successfully");
|
||
|
cy.get("body").type("{esc}");
|
||
|
|
||
|
// verify if the new record is in the stack
|
||
|
verifyKanbanStackCount(7);
|
||
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"PG-13",
|
||
|
"R",
|
||
|
"NC-17",
|
||
|
"Test",
|
||
|
]);
|
||
|
verifyKanbanStackFooterCount([
|
||
|
"0",
|
||
|
"178",
|
||
|
"194",
|
||
|
"223",
|
||
|
"195",
|
||
|
"210",
|
||
|
"1",
|
||
|
]);
|
||
|
|
||
|
mainPage.toggleShowSystemFields();
|
||
|
});
|
||
|
|
||
|
it("Expand record", () => {
|
||
|
// mainPage.toggleShowSystemFields();
|
||
|
// mainPage.showAllColumns();
|
||
|
|
||
|
cy.get(".nc-kanban-stack").eq(1).find(".nc-kanban-item").eq(0).click();
|
||
|
cy.get(".nc-expand-col-Title")
|
||
|
.find(".nc-cell > input")
|
||
|
.then(($el) => {
|
||
|
expect($el[0].value).to.have.string("ACE GOLDFINGER");
|
||
|
});
|
||
|
cy.get("body").type("{esc}");
|
||
|
});
|
||
|
|
||
|
it("Stack context menu- delete stack", () => {
|
||
|
cy.get(".nc-kanban-stack-head").last().scrollIntoView();
|
||
|
cy.get(".nc-kanban-stack-head").last().click();
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-stack-context-menu").should(
|
||
|
"be.visible"
|
||
|
);
|
||
|
cy.getActiveMenu(".nc-dropdown-kanban-stack-context-menu")
|
||
|
.find(".ant-dropdown-menu-item")
|
||
|
.contains("Delete Stack")
|
||
|
.click();
|
||
|
cy.getActiveModal(".nc-modal-kanban-delete-stack").should("be.visible");
|
||
|
cy.getActiveModal(".nc-modal-kanban-delete-stack")
|
||
|
.find(".ant-btn-primary")
|
||
|
.click();
|
||
|
verifyKanbanStackCount(6);
|
||
|
verifyKanbanStackOrder([
|
||
|
"uncategorized",
|
||
|
"G",
|
||
|
"PG",
|
||
|
"PG-13",
|
||
|
"R",
|
||
|
"NC-17",
|
||
|
]);
|
||
|
});
|
||
|
|
||
2 years ago
|
it("Delete Kanban view", () => {
|
||
2 years ago
|
cy.viewDelete(0);
|
||
|
cy.closeTableTab("Film");
|
||
|
});
|
||
|
});
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @copyright Copyright (c) 2021, Xgene Cloud Ltd
|
||
|
*
|
||
|
* @author Pranav C Balan <pranavxc@gmail.com>
|
||
|
* @author Raju Udava <sivadstala@gmail.com>
|
||
|
*
|
||
|
* @license GNU AGPL version 3 or any later version
|
||
|
*
|
||
|
* This program is free software: you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU Affero 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 Affero General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Affero General Public License
|
||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*
|
||
|
*/
|