mirror of
https://github.com/scm-manager/scm-manager.git
synced 2025-11-15 09:46:16 +01:00
merge with 2.0.0-m3
This commit is contained in:
@@ -6,4 +6,8 @@ public abstract class BadRequestException extends ExceptionWithContext {
|
|||||||
public BadRequestException(List<ContextEntry> context, String message) {
|
public BadRequestException(List<ContextEntry> context, String message) {
|
||||||
super(context, message);
|
super(context, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public BadRequestException(List<ContextEntry> context, String message, Exception cause) {
|
||||||
|
super(context, message, cause);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -45,21 +45,24 @@ import java.nio.file.Path;
|
|||||||
public final class InstalledPlugin implements Plugin
|
public final class InstalledPlugin implements Plugin
|
||||||
{
|
{
|
||||||
|
|
||||||
|
public static final String UNINSTALL_MARKER_FILENAME = "uninstall";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructs a new plugin wrapper.
|
* Constructs a new plugin wrapper.
|
||||||
*
|
|
||||||
* @param descriptor wrapped plugin
|
* @param descriptor wrapped plugin
|
||||||
* @param classLoader plugin class loader
|
* @param classLoader plugin class loader
|
||||||
* @param webResourceLoader web resource loader
|
* @param webResourceLoader web resource loader
|
||||||
* @param directory plugin directory
|
* @param directory plugin directory
|
||||||
|
* @param core marked as core or not
|
||||||
*/
|
*/
|
||||||
public InstalledPlugin(InstalledPluginDescriptor descriptor, ClassLoader classLoader,
|
public InstalledPlugin(InstalledPluginDescriptor descriptor, ClassLoader classLoader,
|
||||||
WebResourceLoader webResourceLoader, Path directory)
|
WebResourceLoader webResourceLoader, Path directory, boolean core)
|
||||||
{
|
{
|
||||||
this.descriptor = descriptor;
|
this.descriptor = descriptor;
|
||||||
this.classLoader = classLoader;
|
this.classLoader = classLoader;
|
||||||
this.webResourceLoader = webResourceLoader;
|
this.webResourceLoader = webResourceLoader;
|
||||||
this.directory = directory;
|
this.directory = directory;
|
||||||
|
this.core = core;
|
||||||
}
|
}
|
||||||
|
|
||||||
//~--- get methods ----------------------------------------------------------
|
//~--- get methods ----------------------------------------------------------
|
||||||
@@ -120,7 +123,27 @@ public final class InstalledPlugin implements Plugin
|
|||||||
return webResourceLoader;
|
return webResourceLoader;
|
||||||
}
|
}
|
||||||
|
|
||||||
//~--- fields ---------------------------------------------------------------
|
public boolean isCore() {
|
||||||
|
return core;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isMarkedForUninstall() {
|
||||||
|
return markedForUninstall;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMarkedForUninstall(boolean markedForUninstall) {
|
||||||
|
this.markedForUninstall = markedForUninstall;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isUninstallable() {
|
||||||
|
return uninstallable;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setUninstallable(boolean uninstallable) {
|
||||||
|
this.uninstallable = uninstallable;
|
||||||
|
}
|
||||||
|
|
||||||
|
//~--- fields ---------------------------------------------------------------
|
||||||
|
|
||||||
/** plugin class loader */
|
/** plugin class loader */
|
||||||
private final ClassLoader classLoader;
|
private final ClassLoader classLoader;
|
||||||
@@ -133,4 +156,9 @@ public final class InstalledPlugin implements Plugin
|
|||||||
|
|
||||||
/** plugin web resource loader */
|
/** plugin web resource loader */
|
||||||
private final WebResourceLoader webResourceLoader;
|
private final WebResourceLoader webResourceLoader;
|
||||||
|
|
||||||
|
private final boolean core;
|
||||||
|
|
||||||
|
private boolean markedForUninstall = false;
|
||||||
|
private boolean uninstallable = false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -81,8 +81,16 @@ public interface PluginManager {
|
|||||||
*/
|
*/
|
||||||
void install(String name, boolean restartAfterInstallation);
|
void install(String name, boolean restartAfterInstallation);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Marks the plugin with the given name for uninstall.
|
||||||
|
*
|
||||||
|
* @param name plugin name
|
||||||
|
* @param restartAfterInstallation restart context after plugin has been marked to really uninstall the plugin
|
||||||
|
*/
|
||||||
|
void uninstall(String name, boolean restartAfterInstallation);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Install all pending plugins and restart the scm context.
|
* Install all pending plugins and restart the scm context.
|
||||||
*/
|
*/
|
||||||
void installPendingAndRestart();
|
void executePendingAndRestart();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -122,6 +122,7 @@ public class ModifyCommandRequest implements Resetable, Validateable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void cleanup() {
|
void cleanup() {
|
||||||
|
if (content.exists()) {
|
||||||
try {
|
try {
|
||||||
IOUtil.delete(content);
|
IOUtil.delete(content);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@@ -129,6 +130,7 @@ public class ModifyCommandRequest implements Resetable, Validateable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static class CreateFileRequest extends ContentModificationRequest {
|
public static class CreateFileRequest extends ContentModificationRequest {
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
//@flow
|
//@flow
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { Link } from "react-router-dom";
|
import {Link} from "react-router-dom";
|
||||||
import type { Branch, Repository } from "@scm-manager/ui-types";
|
import type {Branch, Repository} from "@scm-manager/ui-types";
|
||||||
import injectSheet from "react-jss";
|
import injectSheet from "react-jss";
|
||||||
import { ExtensionPoint, binder } from "@scm-manager/ui-extensions";
|
import {binder, ExtensionPoint} from "@scm-manager/ui-extensions";
|
||||||
import {ButtonGroup} from "./buttons";
|
|
||||||
import classNames from "classnames";
|
import classNames from "classnames";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
@@ -64,33 +63,48 @@ class Breadcrumb extends React.Component<Props> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { classes, baseUrl, branch, defaultBranch, branches, revision, path, repository } = this.props;
|
const {
|
||||||
|
classes,
|
||||||
|
baseUrl,
|
||||||
|
branch,
|
||||||
|
defaultBranch,
|
||||||
|
branches,
|
||||||
|
revision,
|
||||||
|
path,
|
||||||
|
repository
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={classes.flexRow}>
|
<div className={classes.flexRow}>
|
||||||
<nav className={classNames(classes.flexStart, "breadcrumb sources-breadcrumb")} aria-label="breadcrumbs">
|
<nav
|
||||||
|
className={classNames(
|
||||||
|
classes.flexStart,
|
||||||
|
"breadcrumb sources-breadcrumb"
|
||||||
|
)}
|
||||||
|
aria-label="breadcrumbs"
|
||||||
|
>
|
||||||
<ul>{this.renderPath()}</ul>
|
<ul>{this.renderPath()}</ul>
|
||||||
</nav>
|
</nav>
|
||||||
{
|
{binder.hasExtension("repos.sources.actionbar") && (
|
||||||
binder.hasExtension("repos.sources.actionbar") &&
|
|
||||||
<div className={classes.buttonGroup}>
|
<div className={classes.buttonGroup}>
|
||||||
<ButtonGroup>
|
|
||||||
<ExtensionPoint
|
<ExtensionPoint
|
||||||
name="repos.sources.actionbar"
|
name="repos.sources.actionbar"
|
||||||
props={{
|
props={{
|
||||||
baseUrl,
|
baseUrl,
|
||||||
branch: branch ? branch : defaultBranch,
|
branch: branch ? branch : defaultBranch,
|
||||||
path,
|
path,
|
||||||
isBranchUrl: branches &&
|
isBranchUrl:
|
||||||
branches.filter(b => b.name.replace("/", "%2F") === revision).length > 0,
|
branches &&
|
||||||
|
branches.filter(
|
||||||
|
b => b.name.replace("/", "%2F") === revision
|
||||||
|
).length > 0,
|
||||||
repository
|
repository
|
||||||
}}
|
}}
|
||||||
renderAll={true}
|
renderAll={true}
|
||||||
/>
|
/>
|
||||||
</ButtonGroup>
|
|
||||||
</div>
|
</div>
|
||||||
}
|
)}
|
||||||
</div>
|
</div>
|
||||||
<hr className={classes.noMargin} />
|
<hr className={classes.noMargin} />
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -48,6 +48,7 @@ type Props = {
|
|||||||
footerRight: React.Node,
|
footerRight: React.Node,
|
||||||
link?: string,
|
link?: string,
|
||||||
action?: () => void,
|
action?: () => void,
|
||||||
|
className?: string,
|
||||||
|
|
||||||
// context props
|
// context props
|
||||||
classes: any
|
classes: any
|
||||||
@@ -72,13 +73,14 @@ class CardColumn extends React.Component<Props> {
|
|||||||
contentRight,
|
contentRight,
|
||||||
footerLeft,
|
footerLeft,
|
||||||
footerRight,
|
footerRight,
|
||||||
classes
|
classes,
|
||||||
|
className
|
||||||
} = this.props;
|
} = this.props;
|
||||||
const link = this.createLink();
|
const link = this.createLink();
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{link}
|
{link}
|
||||||
<article className={classNames("media", classes.inner)}>
|
<article className={classNames("media", className, classes.inner)}>
|
||||||
<figure className={classNames(classes.centerImage, "media-left")}>
|
<figure className={classNames(classes.centerImage, "media-left")}>
|
||||||
{avatar}
|
{avatar}
|
||||||
</figure>
|
</figure>
|
||||||
|
|||||||
@@ -14,3 +14,9 @@ export const isMailValid = (mail: string) => {
|
|||||||
export const isNumberValid = (number: string) => {
|
export const isNumberValid = (number: string) => {
|
||||||
return !isNaN(number);
|
return !isNaN(number);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const pathRegex = /^((?!\/{2,}).)*$/;
|
||||||
|
|
||||||
|
export const isPathValid = (path: string) => {
|
||||||
|
return pathRegex.test(path);
|
||||||
|
};
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
import * as validator from "./validation";
|
import * as validator from "./validation";
|
||||||
|
|
||||||
describe("test name validation", () => {
|
describe("test name validation", () => {
|
||||||
it("should return false", () => {
|
|
||||||
// invalid names taken from ValidationUtilTest.java
|
// invalid names taken from ValidationUtilTest.java
|
||||||
const invalidNames = [
|
const invalidNames = [
|
||||||
"@test",
|
"@test",
|
||||||
@@ -23,11 +22,11 @@ describe("test name validation", () => {
|
|||||||
"!_!"
|
"!_!"
|
||||||
];
|
];
|
||||||
for (let name of invalidNames) {
|
for (let name of invalidNames) {
|
||||||
|
it(`should return false for '${name}'`, () => {
|
||||||
expect(validator.isNameValid(name)).toBe(false);
|
expect(validator.isNameValid(name)).toBe(false);
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
it("should return true", () => {
|
|
||||||
// valid names taken from ValidationUtilTest.java
|
// valid names taken from ValidationUtilTest.java
|
||||||
const validNames = [
|
const validNames = [
|
||||||
"test",
|
"test",
|
||||||
@@ -46,13 +45,13 @@ describe("test name validation", () => {
|
|||||||
"and@this"
|
"and@this"
|
||||||
];
|
];
|
||||||
for (let name of validNames) {
|
for (let name of validNames) {
|
||||||
|
it(`should return true for '${name}'`, () => {
|
||||||
expect(validator.isNameValid(name)).toBe(true);
|
expect(validator.isNameValid(name)).toBe(true);
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("test mail validation", () => {
|
describe("test mail validation", () => {
|
||||||
it("should return false", () => {
|
|
||||||
// invalid taken from ValidationUtilTest.java
|
// invalid taken from ValidationUtilTest.java
|
||||||
const invalid = [
|
const invalid = [
|
||||||
"ostfalia.de",
|
"ostfalia.de",
|
||||||
@@ -63,11 +62,11 @@ describe("test mail validation", () => {
|
|||||||
"s.sdorra@[ostfalia.de"
|
"s.sdorra@[ostfalia.de"
|
||||||
];
|
];
|
||||||
for (let mail of invalid) {
|
for (let mail of invalid) {
|
||||||
|
it(`should return false for '${mail}'`, () => {
|
||||||
expect(validator.isMailValid(mail)).toBe(false);
|
expect(validator.isMailValid(mail)).toBe(false);
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
it("should return true", () => {
|
|
||||||
// valid taken from ValidationUtilTest.java
|
// valid taken from ValidationUtilTest.java
|
||||||
const valid = [
|
const valid = [
|
||||||
"s.sdorra@ostfalia.de",
|
"s.sdorra@ostfalia.de",
|
||||||
@@ -82,22 +81,38 @@ describe("test mail validation", () => {
|
|||||||
"\"S Sdorra\"@scm.solutions"
|
"\"S Sdorra\"@scm.solutions"
|
||||||
];
|
];
|
||||||
for (let mail of valid) {
|
for (let mail of valid) {
|
||||||
|
it(`should return true for '${mail}'`, () => {
|
||||||
expect(validator.isMailValid(mail)).toBe(true);
|
expect(validator.isMailValid(mail)).toBe(true);
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("test number validation", () => {
|
describe("test number validation", () => {
|
||||||
it("should return false", () => {
|
|
||||||
const invalid = ["1a", "35gu", "dj6", "45,5", "test"];
|
const invalid = ["1a", "35gu", "dj6", "45,5", "test"];
|
||||||
for (let number of invalid) {
|
for (let number of invalid) {
|
||||||
|
it(`should return false for '${number}'`, () => {
|
||||||
expect(validator.isNumberValid(number)).toBe(false);
|
expect(validator.isNumberValid(number)).toBe(false);
|
||||||
}
|
|
||||||
});
|
});
|
||||||
it("should return true", () => {
|
}
|
||||||
const valid = ["1", "35", "2", "235", "34.4"];
|
const valid = ["1", "35", "2", "235", "34.4"];
|
||||||
for (let number of valid) {
|
for (let number of valid) {
|
||||||
|
it(`should return true for '${number}'`, () => {
|
||||||
expect(validator.isNumberValid(number)).toBe(true);
|
expect(validator.isNumberValid(number)).toBe(true);
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("test path validation", () => {
|
||||||
|
const invalid = ["//", "some//path", "end//"];
|
||||||
|
for (let path of invalid) {
|
||||||
|
it(`should return false for '${path}'`, () => {
|
||||||
|
expect(validator.isPathValid(path)).toBe(false);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
const valid = ["", "/", "dir", "some/path", "end/"];
|
||||||
|
for (let path of valid) {
|
||||||
|
it(`should return true for '${path}'`, () => {
|
||||||
|
expect(validator.isPathValid(path)).toBe(true);
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,12 +4,14 @@ import type {Collection, Links} from "./hal";
|
|||||||
export type Plugin = {
|
export type Plugin = {
|
||||||
name: string,
|
name: string,
|
||||||
version: string,
|
version: string,
|
||||||
|
newVersion?: string,
|
||||||
displayName: string,
|
displayName: string,
|
||||||
description?: string,
|
description?: string,
|
||||||
author: string,
|
author: string,
|
||||||
category: string,
|
category: string,
|
||||||
avatarUrl: string,
|
avatarUrl: string,
|
||||||
pending: boolean,
|
pending: boolean,
|
||||||
|
markedForUninstall?: boolean,
|
||||||
dependencies: string[],
|
dependencies: string[],
|
||||||
_links: Links
|
_links: Links
|
||||||
};
|
};
|
||||||
@@ -24,3 +26,12 @@ export type PluginGroup = {
|
|||||||
name: string,
|
name: string,
|
||||||
plugins: Plugin[]
|
plugins: Plugin[]
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type PendingPlugins = {
|
||||||
|
_links: Links,
|
||||||
|
_embedded: {
|
||||||
|
new: [],
|
||||||
|
update: [],
|
||||||
|
uninstall: []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ export type { SubRepository, File } from "./Sources";
|
|||||||
|
|
||||||
export type { SelectValue, AutocompleteObject } from "./Autocomplete";
|
export type { SelectValue, AutocompleteObject } from "./Autocomplete";
|
||||||
|
|
||||||
export type { Plugin, PluginCollection, PluginGroup } from "./Plugin";
|
export type { Plugin, PluginCollection, PluginGroup, PendingPlugins } from "./Plugin";
|
||||||
|
|
||||||
export type { RepositoryRole } from "./RepositoryRole";
|
export type { RepositoryRole } from "./RepositoryRole";
|
||||||
|
|
||||||
|
|||||||
@@ -29,22 +29,36 @@
|
|||||||
"installedNavLink": "Installiert",
|
"installedNavLink": "Installiert",
|
||||||
"availableNavLink": "Verfügbar"
|
"availableNavLink": "Verfügbar"
|
||||||
},
|
},
|
||||||
"installPending": "Austehende Plugins installieren",
|
"executePending": "Ausstehende Plugin-Änderungen ausführen",
|
||||||
"noPlugins": "Keine Plugins gefunden.",
|
"noPlugins": "Keine Plugins gefunden.",
|
||||||
"modal": {
|
"modal": {
|
||||||
"title": "{{name}} Plugin installieren",
|
"title": {
|
||||||
"restart": "Neustarten um Plugin zu aktivieren",
|
"install": "{{name}} Plugin installieren",
|
||||||
|
"update": "{{name}} Plugin aktualisieren",
|
||||||
|
"uninstall": "{{name}} Plugin deinstallieren"
|
||||||
|
},
|
||||||
|
"restart": "Neustarten, um Plugin-Änderungen wirksam zu machen",
|
||||||
"install": "Installieren",
|
"install": "Installieren",
|
||||||
|
"update": "Aktualisieren",
|
||||||
|
"uninstall": "Deinstallieren",
|
||||||
|
"installQueue": "Werden installiert:",
|
||||||
|
"updateQueue": "Werden aktualisiert:",
|
||||||
|
"uninstallQueue": "Werden deinstalliert:",
|
||||||
"installAndRestart": "Installieren und Neustarten",
|
"installAndRestart": "Installieren und Neustarten",
|
||||||
|
"updateAndRestart": "Aktualisieren und Neustarten",
|
||||||
|
"uninstallAndRestart": "Deinstallieren and Neustarten",
|
||||||
|
"executeAndRestart": "Ausführen und Neustarten",
|
||||||
"abort": "Abbrechen",
|
"abort": "Abbrechen",
|
||||||
"author": "Autor",
|
"author": "Autor",
|
||||||
"version": "Version",
|
"version": "Version",
|
||||||
|
"currentVersion": "Installierte Version",
|
||||||
|
"newVersion": "Neue Version",
|
||||||
"dependencyNotification": "Mit diesem Plugin werden folgende Abhängigkeiten mit installiert, wenn sie noch nicht vorhanden sind!",
|
"dependencyNotification": "Mit diesem Plugin werden folgende Abhängigkeiten mit installiert, wenn sie noch nicht vorhanden sind!",
|
||||||
"dependencies": "Abhängigkeiten",
|
"dependencies": "Abhängigkeiten",
|
||||||
"successNotification": "Das Plugin wurde erfolgreich installiert. Um Änderungen an der UI zu sehen, muss die Seite neu geladen werden:",
|
"successNotification": "Das Plugin wurde erfolgreich installiert. Um Änderungen an der UI zu sehen, muss die Seite neu geladen werden:",
|
||||||
"reload": "jetzt neu laden",
|
"reload": "jetzt neu laden",
|
||||||
"restartNotification": "Der SCM-Manager Kontext sollte nur neu gestartet werden, wenn aktuell niemand damit arbeitet.",
|
"restartNotification": "Der SCM-Manager Kontext sollte nur neu gestartet werden, wenn aktuell niemand damit arbeitet.",
|
||||||
"installPending": "Die folgenden Plugins werden installiert. Anschließend wird der SCM-Manager Kontext neu gestartet."
|
"executePending": "Die folgenden Plugin-Änderungen werden ausgeführt. Anschließend wird der SCM-Manager Kontext neu gestartet."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"repositoryRole": {
|
"repositoryRole": {
|
||||||
|
|||||||
@@ -29,22 +29,36 @@
|
|||||||
"installedNavLink": "Installed",
|
"installedNavLink": "Installed",
|
||||||
"availableNavLink": "Available"
|
"availableNavLink": "Available"
|
||||||
},
|
},
|
||||||
"installPending": "Install pending plugins",
|
"executePending": "Execute pending plugin changes",
|
||||||
"noPlugins": "No plugins found.",
|
"noPlugins": "No plugins found.",
|
||||||
"modal": {
|
"modal": {
|
||||||
"title": "Install {{name}} Plugin",
|
"title": {
|
||||||
"restart": "Restart to activate",
|
"install": "Install {{name}} Plugin",
|
||||||
|
"update": "Update {{name}} Plugin",
|
||||||
|
"uninstall": "Uninstall {{name}} Plugin"
|
||||||
|
},
|
||||||
|
"restart": "Restart to make plugin changes effective",
|
||||||
"install": "Install",
|
"install": "Install",
|
||||||
|
"update": "Update",
|
||||||
|
"uninstall": "Uninstall",
|
||||||
|
"installQueue": "Will be installed:",
|
||||||
|
"updateQueue": "Will be updated:",
|
||||||
|
"uninstallQueue": "Will be uninstalled:",
|
||||||
"installAndRestart": "Install and Restart",
|
"installAndRestart": "Install and Restart",
|
||||||
|
"updateAndRestart": "Update and Restart",
|
||||||
|
"uninstallAndRestart": "Uninstall and Restart",
|
||||||
|
"executeAndRestart": "Execute and Restart",
|
||||||
"abort": "Abort",
|
"abort": "Abort",
|
||||||
"author": "Author",
|
"author": "Author",
|
||||||
"version": "Version",
|
"version": "Version",
|
||||||
|
"currentVersion": "Installed version",
|
||||||
|
"newVersion": "New version",
|
||||||
"dependencyNotification": "With this plugin, the following dependencies will be installed if they are not available yet!",
|
"dependencyNotification": "With this plugin, the following dependencies will be installed if they are not available yet!",
|
||||||
"dependencies": "Dependencies",
|
"dependencies": "Dependencies",
|
||||||
"successNotification": "Successful installed plugin. You have to reload the page, to see ui changes:",
|
"successNotification": "Successful installed plugin. You have to reload the page, to see ui changes:",
|
||||||
"reload": "reload now",
|
"reload": "reload now",
|
||||||
"restartNotification": "You should only restart the scm-manager context if no one else is currently working with it.",
|
"restartNotification": "You should only restart the scm-manager context if no one else is currently working with it.",
|
||||||
"installPending": "The following plugins will be installed and after installation the scm-manager context will be restarted."
|
"executePending": "The following plugin changes will be executed and after that the scm-manager context will be restarted."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"repositoryRole": {
|
"repositoryRole": {
|
||||||
|
|||||||
83
scm-ui/public/locales/es/admin.json
Normal file
83
scm-ui/public/locales/es/admin.json
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
{
|
||||||
|
"admin": {
|
||||||
|
"menu": {
|
||||||
|
"navigationLabel": "Menú de administración",
|
||||||
|
"informationNavLink": "Información",
|
||||||
|
"settingsNavLink": "Ajustes",
|
||||||
|
"generalNavLink": "General"
|
||||||
|
},
|
||||||
|
"info": {
|
||||||
|
"currentAppVersion": "Versión actual de la aplicación",
|
||||||
|
"communityTitle": "Soporte de la comunidad",
|
||||||
|
"communityIconAlt": "Icono del soporte de la comunidad",
|
||||||
|
"communityInfo": "Contacte con el equipo de soporte de SCM-Manager para questiones acerca de SCM-Manager, para informar de errores o pedir nuevas funcionalidades use los canales oficiales.",
|
||||||
|
"communityButton": "Contactar con nuestro equipo",
|
||||||
|
"enterpriseTitle": "Soporte empresarial",
|
||||||
|
"enterpriseIconAlt": "Icono del soporte para empresas",
|
||||||
|
"enterpriseInfo": "¿Necesita ayuda para la integración de SMC-Manager en sus procesos, con la personalización de la herramienta o simplemente un acuerdo de nivel de servicio (SLA)?",
|
||||||
|
"enterprisePartner": "Póngase en contacto con nuestro socio de desarrollo Cloudogu! Su equipo está esperando para tratar sus requisitos con usted y estará encantado de darle un presupuesto.",
|
||||||
|
"enterpriseLink": "https://cloudogu.com/en/scm-manager-enterprise/",
|
||||||
|
"enterpriseButton": "Pedir soporte empresarial"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"plugins": {
|
||||||
|
"title": "Complementos",
|
||||||
|
"installedSubtitle": "Complementos instalados",
|
||||||
|
"availableSubtitle": "Complementos disponibles",
|
||||||
|
"menu": {
|
||||||
|
"pluginsNavLink": "Complementos",
|
||||||
|
"installedNavLink": "Instalados",
|
||||||
|
"availableNavLink": "Disponibles"
|
||||||
|
},
|
||||||
|
"installPending": "Instalar los complementos pendientes",
|
||||||
|
"noPlugins": "No se han encontrado complementos.",
|
||||||
|
"modal": {
|
||||||
|
"title": "Instalar complemento {{name}} ",
|
||||||
|
"restart": "Reiniciar para activar",
|
||||||
|
"install": "Instalar",
|
||||||
|
"installAndRestart": "Instalar y reiniciar",
|
||||||
|
"abort": "Cancelar",
|
||||||
|
"author": "Autor",
|
||||||
|
"version": "Versión",
|
||||||
|
"dependencyNotification": "Con este complemento las siguientes dependencias serán instaladas si no lo han sido ya",
|
||||||
|
"dependencies": "Dependencias",
|
||||||
|
"successNotification": "Complemento instalado correctamente. Necesita recargar la página para ver los cambios:",
|
||||||
|
"reload": "recargar ahora",
|
||||||
|
"restartNotification": "Usted debería reiniciar scm-manager sólo si actualmente no hay nadie trabajando con el.",
|
||||||
|
"installPending": "Los siguientes complementos serán instalados y después de la instalación sdm-manager será reiniciado."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"repositoryRole": {
|
||||||
|
"navLink": "Roles y permisos",
|
||||||
|
"title": "Roles y permisos",
|
||||||
|
"errorTitle": "Error",
|
||||||
|
"errorSubtitle": "Error desconocido",
|
||||||
|
"createSubtitle": "Crear nuevo rol",
|
||||||
|
"editSubtitle": "Editar rol",
|
||||||
|
"overview": {
|
||||||
|
"title": "Visión general de todos los roles",
|
||||||
|
"noPermissionRoles": "No se han encontrado roles.",
|
||||||
|
"createButton": "Crear rol"
|
||||||
|
},
|
||||||
|
"editButton": "Editar",
|
||||||
|
"name": "Nombre",
|
||||||
|
"type": "Tipo",
|
||||||
|
"verbs": "Permisos",
|
||||||
|
"system": "Sistema",
|
||||||
|
"form": {
|
||||||
|
"name": "Nombre",
|
||||||
|
"permissions": "Permisos",
|
||||||
|
"submit": "Guardar"
|
||||||
|
},
|
||||||
|
"delete": {
|
||||||
|
"button": "Borrar",
|
||||||
|
"subtitle": "Eliminar el rol",
|
||||||
|
"confirmAlert": {
|
||||||
|
"title": "Eliminar el rol",
|
||||||
|
"message": "¿Realmente desea borrar el rol? Todos los usuarios de este rol perderń sus permisos.",
|
||||||
|
"submit": "Sí",
|
||||||
|
"cancel": "No"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
90
scm-ui/public/locales/es/commons.json
Normal file
90
scm-ui/public/locales/es/commons.json
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
{
|
||||||
|
"login": {
|
||||||
|
"title": "Iniciar sesión",
|
||||||
|
"subtitle": "Por favor inicie sesión para continuar",
|
||||||
|
"logo-alt": "SCM-Manager",
|
||||||
|
"username-placeholder": "Su nombre de usuario",
|
||||||
|
"password-placeholder": "Su contraseña",
|
||||||
|
"submit": "Iniciar sesión",
|
||||||
|
"plugin": "Plugin",
|
||||||
|
"feature": "Feature",
|
||||||
|
"tip": "Tip",
|
||||||
|
"loading": "Cargando ...",
|
||||||
|
"error": "Error"
|
||||||
|
},
|
||||||
|
"logout": {
|
||||||
|
"error": {
|
||||||
|
"title": "Cierre de sesión fallido",
|
||||||
|
"subtitle": "Ha ocurrido un error al cerrar la sesión"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"app": {
|
||||||
|
"error": {
|
||||||
|
"title": "Error",
|
||||||
|
"subtitle": "Ha ocurrido un error desconocido"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"errorNotification": {
|
||||||
|
"prefix": "Error",
|
||||||
|
"loginLink": "Aquí puede iniciar la sesión de nuevo.",
|
||||||
|
"timeout": "La sesión ha caducado",
|
||||||
|
"wrongLoginCredentials": "Credenciales incorrectas",
|
||||||
|
"forbidden": "Usted no tiene permiso para ver esta sección"
|
||||||
|
},
|
||||||
|
"loading": {
|
||||||
|
"alt": "Cargando ..."
|
||||||
|
},
|
||||||
|
"logo": {
|
||||||
|
"alt": "SCM-Manager"
|
||||||
|
},
|
||||||
|
"primary-navigation": {
|
||||||
|
"repositories": "Repositorios",
|
||||||
|
"users": "Usuarios",
|
||||||
|
"logout": "Cerrar sesión",
|
||||||
|
"groups": "Grupos",
|
||||||
|
"admin": "Administración"
|
||||||
|
},
|
||||||
|
"filterEntries": "Filtrar entradas",
|
||||||
|
"autocomplete": {
|
||||||
|
"group": "Grupo",
|
||||||
|
"user": "Usuario",
|
||||||
|
"noGroupOptions": "No hay sugerencias disponibles",
|
||||||
|
"groupPlaceholder": "Nombre del grupo",
|
||||||
|
"noUserOptions": "No hay sugerencias disponibles",
|
||||||
|
"userPlaceholder": "Nombre de usuario",
|
||||||
|
"loading": "Cargando..."
|
||||||
|
},
|
||||||
|
"paginator": {
|
||||||
|
"next": "Siguiente",
|
||||||
|
"previous": "Anterior"
|
||||||
|
},
|
||||||
|
"profile": {
|
||||||
|
"navigationLabel": "Menú de sección",
|
||||||
|
"informationNavLink": "Información",
|
||||||
|
"changePasswordNavLink": "Cambiar contraseña",
|
||||||
|
"settingsNavLink": "Ajustes",
|
||||||
|
"username": "Nombre de usuario",
|
||||||
|
"displayName": "Nombre a mostrar",
|
||||||
|
"mail": "Correo electrónico",
|
||||||
|
"groups": "Grupos",
|
||||||
|
"information": "Información",
|
||||||
|
"change-password": "Cambiar contraseña",
|
||||||
|
"error-title": "Error",
|
||||||
|
"error-subtitle": "No se puede mostrar la sección",
|
||||||
|
"error": "Error",
|
||||||
|
"error-message": "'me' no está definido"
|
||||||
|
},
|
||||||
|
"password": {
|
||||||
|
"label": "Contraseña",
|
||||||
|
"newPassword": "Nueva contraseña",
|
||||||
|
"passwordHelpText": "Contraseña del usuario en texto plano",
|
||||||
|
"passwordConfirmHelpText": "Repita la contraseña para confirmar",
|
||||||
|
"currentPassword": "Contraseña actual",
|
||||||
|
"currentPasswordHelpText": "La contraseña ya está en uso",
|
||||||
|
"confirmPassword": "Confirme la contraseña",
|
||||||
|
"passwordInvalid": "La contraseña debe tener entre 6 y 32 caracteres",
|
||||||
|
"passwordConfirmFailed": "Las contraseñas deben ser identicas",
|
||||||
|
"submit": "Guardar",
|
||||||
|
"changedSuccessfully": "Contraseña cambiada correctamente"
|
||||||
|
}
|
||||||
|
}
|
||||||
78
scm-ui/public/locales/es/config.json
Normal file
78
scm-ui/public/locales/es/config.json
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
{
|
||||||
|
"config": {
|
||||||
|
"navigationLabel": "Menú de administración",
|
||||||
|
"title": "Configuración global",
|
||||||
|
"errorTitle": "Error",
|
||||||
|
"errorSubtitle": "Error de configuración desconocido",
|
||||||
|
"form": {
|
||||||
|
"submit": "Enviar",
|
||||||
|
"submit-success-notification": "¡Configuración cambiada correctamente!",
|
||||||
|
"no-read-permission-notification": "Por favor, tenga en cuenta: ¡No tiene permiso para ver la configuración!",
|
||||||
|
"no-write-permission-notification": "Por favor, tenga en cuenta: ¡No tiene permiso para editar la configuración!"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"proxy-settings": {
|
||||||
|
"name": "Ajustes del proxy",
|
||||||
|
"proxy-password": "Contraseña del proxy",
|
||||||
|
"proxy-port": "Puerto del proxy",
|
||||||
|
"proxy-server": "Servidor proxy",
|
||||||
|
"proxy-user": "Usuario del proxy",
|
||||||
|
"enable-proxy": "Habilitar proxy",
|
||||||
|
"proxy-excludes": "Excepciones del proxy",
|
||||||
|
"remove-proxy-exclude-button": "Eliminar las excepciones del proxy",
|
||||||
|
"add-proxy-exclude-error": "La excepción que desea añadir al proxy es incorrecta",
|
||||||
|
"add-proxy-exclude-textfield": "Añada aquí las excepciones que desee incluir al proxy",
|
||||||
|
"add-proxy-exclude-button": "Añadir excepción al proxy"
|
||||||
|
},
|
||||||
|
"base-url-settings": {
|
||||||
|
"name": "Ajustes de la URL base",
|
||||||
|
"base-url": "URL base",
|
||||||
|
"force-base-url": "Forzar la URL base"
|
||||||
|
},
|
||||||
|
"login-attempt": {
|
||||||
|
"name": "Intento de inicio de sesión",
|
||||||
|
"login-attempt-limit": "Límite de intentos de inicio de sesión",
|
||||||
|
"login-attempt-limit-timeout": "Tiempo de espera para el intento de inicio de sesión"
|
||||||
|
},
|
||||||
|
"general-settings": {
|
||||||
|
"realm-description": "Descripción del dominio",
|
||||||
|
"disable-grouping-grid": "Deshabilitar grupos",
|
||||||
|
"date-format": "Formato de la fecha",
|
||||||
|
"anonymous-access-enabled": "Acceso anónimo habilitado",
|
||||||
|
"skip-failed-authenticators": "Omitir autenticadores fallidos",
|
||||||
|
"plugin-url": "URL del almacén de complementos",
|
||||||
|
"enabled-xsrf-protection": "Protección XSRF habilitada",
|
||||||
|
"namespace-strategy": "Estrategia para el espacio de nombres",
|
||||||
|
"login-info-url": "URL de información de inicio de sesión"
|
||||||
|
},
|
||||||
|
"validation": {
|
||||||
|
"date-format-invalid": "El formato de la fecha es incorrecto",
|
||||||
|
"login-attempt-limit-timeout-invalid": "El valor no es un número",
|
||||||
|
"login-attempt-limit-invalid": "El valor no es un número",
|
||||||
|
"plugin-url-invalid": "La URL es incorrecta"
|
||||||
|
},
|
||||||
|
"help": {
|
||||||
|
"realmDescriptionHelpText": "Descripción del dominio de autenticación.",
|
||||||
|
"dateFormatHelpText": "Formato de la fecha. Por favor, heche un vistazo a la documentación de MomentJS.",
|
||||||
|
"pluginUrlHelpText": "La URL de la API del almacén de complementos. Explicación de los marcadores: version = Versión de SCM-Manager; os = Sistema operativo; arch = Arquitectura",
|
||||||
|
"enableForwardingHelpText": "Habilitar el redireccionamiento de puertos para mod_proxy.",
|
||||||
|
"disableGroupingGridHelpText": "Deshabilitar los grupos de repositorios. Se requiere una recarga completa de la página después de un cambio en este valor.",
|
||||||
|
"allowAnonymousAccessHelpText": "Los usuarios anónimos tienen acceso de lectura en los repositorios públicos.",
|
||||||
|
"skipFailedAuthenticatorsHelpText": "No detenga la cadena de autenticación si un autenticador encuentra al usuario pero no puede autenticarlo.",
|
||||||
|
"adminGroupsHelpText": "Nombres de los grupos con permisos de administrador.",
|
||||||
|
"adminUsersHelpText": "Nombres de los usuarios con permisos de administrador.",
|
||||||
|
"forceBaseUrlHelpText": "Redirige a la URL base si la solicitud proviene de otra URL.",
|
||||||
|
"baseUrlHelpText": "La URL de la aplicación (con la ruta del contexto), por ejemplo: http://localhost:8080/scm",
|
||||||
|
"loginAttemptLimitHelpText": "Máximo número permitido de intentos de inicio de sesión. Use -1 para deshabilitar este límite.",
|
||||||
|
"loginAttemptLimitTimeoutHelpText": "Tiempo de espera en segundos para los usuarios que están deshabilitados temporalmente debido a demasiado intentos fallidos de inicio de sesión.",
|
||||||
|
"enableProxyHelpText": "Habilitar proxy",
|
||||||
|
"proxyPortHelpText": "El puerto del proxy",
|
||||||
|
"proxyPasswordHelpText": "La contraseña para la autenticación del servidor proxy.",
|
||||||
|
"proxyServerHelpText": "El servidor proxy",
|
||||||
|
"proxyUserHelpText": "El nombre de usuario para la autenticación del servidor proxy.",
|
||||||
|
"proxyExcludesHelpText": "Patrones globales para hostnames que deben excluirse de la configuración del proxy.",
|
||||||
|
"enableXsrfProtectionHelpText": "Habilitar la protección de cookies XSRF. Nota: Esta funcionalidad todavía es experimental.",
|
||||||
|
"nameSpaceStrategyHelpText": "La estrategia para el espacio de nombres.",
|
||||||
|
"loginInfoUrlHelpText": "URL para la información en el inicio de sesión (consejos sobre complementos y funcionalidades en la página de inicio de sesión). Si esto se omite, no se mostrará información de inicio de sesión."
|
||||||
|
}
|
||||||
|
}
|
||||||
76
scm-ui/public/locales/es/groups.json
Normal file
76
scm-ui/public/locales/es/groups.json
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
{
|
||||||
|
"group": {
|
||||||
|
"name": "Nombre",
|
||||||
|
"description": "Descripción",
|
||||||
|
"creationDate": "Fecha de creación",
|
||||||
|
"lastModified": "Última modificación",
|
||||||
|
"type": "Tipo",
|
||||||
|
"external": "Externo",
|
||||||
|
"internal": "Interno",
|
||||||
|
"members": "Miembros"
|
||||||
|
},
|
||||||
|
"groups": {
|
||||||
|
"title": "Grupos",
|
||||||
|
"subtitle": "Crear, leer, actualizar y borrar grupos",
|
||||||
|
"noGroups": "No se han encontrado grupos."
|
||||||
|
},
|
||||||
|
"singleGroup": {
|
||||||
|
"errorTitle": "Error",
|
||||||
|
"errorSubtitle": "Error de grupo desconocido",
|
||||||
|
"menu": {
|
||||||
|
"navigationLabel": "Menú de grupo",
|
||||||
|
"informationNavLink": "Información",
|
||||||
|
"settingsNavLink": "Ajustes",
|
||||||
|
"generalNavLink": "General",
|
||||||
|
"setPermissionsNavLink": "Permisos"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"add-group": {
|
||||||
|
"title": "Crear grupo",
|
||||||
|
"subtitle": "Crear un nuevo grupo"
|
||||||
|
},
|
||||||
|
"create-group-button": {
|
||||||
|
"label": "Crear grupo"
|
||||||
|
},
|
||||||
|
"edit-group-button": {
|
||||||
|
"label": "Editar"
|
||||||
|
},
|
||||||
|
"add-member-button": {
|
||||||
|
"label": "Añadir miembro"
|
||||||
|
},
|
||||||
|
"remove-member-button": {
|
||||||
|
"label": "Eliminar miembro"
|
||||||
|
},
|
||||||
|
"add-member-textfield": {
|
||||||
|
"label": "Añadir miembro",
|
||||||
|
"error": "El nombre del miembro es incorrecto"
|
||||||
|
},
|
||||||
|
"add-member-autocomplete": {
|
||||||
|
"placeholder": "Introducir el nombre del miembro",
|
||||||
|
"loading": "Cargando...",
|
||||||
|
"no-options": "No hay sugerencias disponibles"
|
||||||
|
},
|
||||||
|
"groupForm": {
|
||||||
|
"subtitle": "Editar grupo",
|
||||||
|
"externalSubtitle": "Editar grupo externo",
|
||||||
|
"submit": "Guardar",
|
||||||
|
"nameError": "El nombre del grupo es incorrecto",
|
||||||
|
"descriptionError": "La descripción es incorrecta",
|
||||||
|
"help": {
|
||||||
|
"nameHelpText": "Nombre único del grupo",
|
||||||
|
"descriptionHelpText": "Descripción breve del grupo",
|
||||||
|
"memberHelpText": "Nombres de usuario de los miembros del grupo",
|
||||||
|
"externalHelpText": "Los miembros son gestionados por un sistema externo como por ejemplo LDAP"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"deleteGroup": {
|
||||||
|
"subtitle": "Borrar grupo",
|
||||||
|
"button": "Borrar",
|
||||||
|
"confirmAlert": {
|
||||||
|
"title": "Borrar grupo",
|
||||||
|
"message": "¿Realmente desea borrar el grupo?",
|
||||||
|
"submit": "Sí",
|
||||||
|
"cancel": "No"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
6
scm-ui/public/locales/es/permissions.json
Normal file
6
scm-ui/public/locales/es/permissions.json
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"setPermissions": {
|
||||||
|
"button": "Guardar",
|
||||||
|
"setPermissionsSuccessful": "Permisos guardados correctamente"
|
||||||
|
}
|
||||||
|
}
|
||||||
189
scm-ui/public/locales/es/repos.json
Normal file
189
scm-ui/public/locales/es/repos.json
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
{
|
||||||
|
"repository": {
|
||||||
|
"namespace": "Espacio de nombres",
|
||||||
|
"name": "Nombre",
|
||||||
|
"type": "Tipo",
|
||||||
|
"contact": "Contacto",
|
||||||
|
"description": "Descripción",
|
||||||
|
"creationDate": "Fecha de creación",
|
||||||
|
"lastModified": "Última modificación"
|
||||||
|
},
|
||||||
|
"validation": {
|
||||||
|
"namespace-invalid": "El espacio de nombres del repositorio es incorrecto",
|
||||||
|
"name-invalid": "El nombre del repositorio es incorrecto",
|
||||||
|
"contact-invalid": "El contacto debe ser una dirección de correo electrónico válida",
|
||||||
|
"branch": {
|
||||||
|
"nameInvalid": "El nombre de la rama es incorrecto"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"help": {
|
||||||
|
"namespaceHelpText": "El espacio de nombres del repositorio. Este nombre formará parte de la URL del repositorio.",
|
||||||
|
"nameHelpText": "El nombre del repositorio. Este nombre formará parte de la URL del repositorio.",
|
||||||
|
"typeHelpText": "El tipo del repositorio (Mercurial, Git or Subversion).",
|
||||||
|
"contactHelpText": "Dirección del correo electrónico de la persona responsable del repositorio.",
|
||||||
|
"descriptionHelpText": "Breve descripción del repositorio."
|
||||||
|
},
|
||||||
|
"repositoryRoot": {
|
||||||
|
"errorTitle": "Error",
|
||||||
|
"errorSubtitle": "Error de repositorio desconocido",
|
||||||
|
"menu": {
|
||||||
|
"navigationLabel": "Menú de repositorio",
|
||||||
|
"informationNavLink": "Información",
|
||||||
|
"branchesNavLink": "Ramas",
|
||||||
|
"historyNavLink": "Commits",
|
||||||
|
"sourcesNavLink": "Fuentes",
|
||||||
|
"settingsNavLink": "Ajustes",
|
||||||
|
"generalNavLink": "General",
|
||||||
|
"permissionsNavLink": "Permisos"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"overview": {
|
||||||
|
"title": "Repositorios",
|
||||||
|
"subtitle": "Visión general de los repositorios disponibles",
|
||||||
|
"noRepositories": "No se han encontrado repositorios.",
|
||||||
|
"createButton": "Crear repositorio"
|
||||||
|
},
|
||||||
|
"create": {
|
||||||
|
"title": "Crear repositorio",
|
||||||
|
"subtitle": "Crear un nuevo repositorio"
|
||||||
|
},
|
||||||
|
"branches": {
|
||||||
|
"overview": {
|
||||||
|
"title": "Vivisón general de todas las ramas",
|
||||||
|
"noBranches": "No se han encontrado ramas.",
|
||||||
|
"createButton": "Crear rama"
|
||||||
|
},
|
||||||
|
"table": {
|
||||||
|
"branches": "Ramas"
|
||||||
|
},
|
||||||
|
"create": {
|
||||||
|
"title": "Crear rama",
|
||||||
|
"source": "Rama padre",
|
||||||
|
"name": "Nombre",
|
||||||
|
"submit": "Crear rama"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"branch": {
|
||||||
|
"name": "Nombre:",
|
||||||
|
"commits": "Commits",
|
||||||
|
"sources": "Fuentes",
|
||||||
|
"defaultTag": "Por defecto"
|
||||||
|
},
|
||||||
|
"changesets": {
|
||||||
|
"errorTitle": "Error",
|
||||||
|
"errorSubtitle": "No se han podido recuperar los changesets",
|
||||||
|
"noChangesets": "No se han encontrado changesets para esta rama branch.",
|
||||||
|
"branchSelectorLabel": "Ramas"
|
||||||
|
},
|
||||||
|
"changeset": {
|
||||||
|
"description": "Descripción",
|
||||||
|
"summary": "El changeset {{id}} fue entregado {{time}}",
|
||||||
|
"shortSummary": "Entregado {{id}} {{time}}",
|
||||||
|
"tags": "Etiquetas",
|
||||||
|
"diffNotSupported": "La comparación de changesets no es soportada por el tipo de repositorio",
|
||||||
|
"author": {
|
||||||
|
"prefix": "Creado por",
|
||||||
|
"mailto": "Enviar correo electrónico a"
|
||||||
|
},
|
||||||
|
"buttons": {
|
||||||
|
"details": "Detalles",
|
||||||
|
"sources": "Fuentes"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"repositoryForm": {
|
||||||
|
"subtitle": "Editar repositorio",
|
||||||
|
"submit": "Guardar"
|
||||||
|
},
|
||||||
|
"sources": {
|
||||||
|
"file-tree": {
|
||||||
|
"name": "Nombre",
|
||||||
|
"length": "Longitud",
|
||||||
|
"lastModified": "Última modificación",
|
||||||
|
"description": "Descripción",
|
||||||
|
"branch": "Rama"
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"historyButton": "Historia",
|
||||||
|
"sourcesButton": "Fuentes",
|
||||||
|
"downloadButton": "Descargar",
|
||||||
|
"path": "Ruta",
|
||||||
|
"branch": "Rama",
|
||||||
|
"lastModified": "Última modificación",
|
||||||
|
"description": "Discripción",
|
||||||
|
"size": "tamaño"
|
||||||
|
},
|
||||||
|
"noSources": "No se han encontrado fuentes para esta rama."
|
||||||
|
},
|
||||||
|
"permission": {
|
||||||
|
"title": "Editar permisos",
|
||||||
|
"user": "Usuario",
|
||||||
|
"group": "Grupo",
|
||||||
|
"error-title": "Error",
|
||||||
|
"error-subtitle": "Error de permisos desconocido",
|
||||||
|
"name": "Usuario o grupo",
|
||||||
|
"role": "Rol",
|
||||||
|
"custom": "Personalizar",
|
||||||
|
"permissions": "Permisos",
|
||||||
|
"group-permission": "Permiso de grupo",
|
||||||
|
"user-permission": "Permiso de usuario",
|
||||||
|
"edit-permission": {
|
||||||
|
"delete-button": "Borrar",
|
||||||
|
"save-button": "Guardar cambios"
|
||||||
|
},
|
||||||
|
"advanced-button": {
|
||||||
|
"label": "Avanzado"
|
||||||
|
},
|
||||||
|
"delete-permission-button": {
|
||||||
|
"label": "Borrar",
|
||||||
|
"confirm-alert": {
|
||||||
|
"title": "Borrar permiso",
|
||||||
|
"message": "¿Realmente desea borrar el permiso?",
|
||||||
|
"submit": "Sí",
|
||||||
|
"cancel": "No"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"add-permission": {
|
||||||
|
"add-permission-heading": "Añadir nuevo permiso",
|
||||||
|
"submit-button": "Guardar",
|
||||||
|
"name-input-invalid": "¡No se permiten permisos vacíos! ¡Si el permiso no está vacío, su nombre es inválido o ya existe!"
|
||||||
|
},
|
||||||
|
"help": {
|
||||||
|
"groupPermissionHelpText": "Establece si un permiso es de grupo. Si no está marcado es un permiso de usuario.",
|
||||||
|
"nameHelpText": "Gestionar los permisos de un usuario o grupo.",
|
||||||
|
"roleHelpText": "READ = leer; WRITE = leer and escribir; OWNER = leer, escribir y también la capacidad de gestionar las propiedades y permisos. Si no hay nada seleccionado use el botón 'Avanzado' para ver los permisos en detalle.",
|
||||||
|
"permissionsHelpText": "Use esto para especificar su propio conjunto de permisos independientemente de los roles predefinidos."
|
||||||
|
},
|
||||||
|
"advanced": {
|
||||||
|
"dialog": {
|
||||||
|
"title": "Permisos avanzados",
|
||||||
|
"submit": "Guardar",
|
||||||
|
"abort": "Cancelar"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"deleteRepo": {
|
||||||
|
"subtitle": "Borrar repositorio",
|
||||||
|
"button": "Borrar",
|
||||||
|
"confirmAlert": {
|
||||||
|
"title": "Borrar repositorio",
|
||||||
|
"message": "¿Realmente desea borrar el repositorio?",
|
||||||
|
"submit": "sí",
|
||||||
|
"cancel": "No"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"diff": {
|
||||||
|
"changes": {
|
||||||
|
"add": "añadido",
|
||||||
|
"delete": "borrado",
|
||||||
|
"modify": "modificado",
|
||||||
|
"rename": "renombrado",
|
||||||
|
"copy": "copiado"
|
||||||
|
},
|
||||||
|
"sideBySide": "dos columnas",
|
||||||
|
"combined": "combinado"
|
||||||
|
},
|
||||||
|
"fileUpload": {
|
||||||
|
"clickHere": "Haga click aquí para seleccionar su fichero",
|
||||||
|
"dragAndDrop": "Arrastre y suelte los ficheros aquí"
|
||||||
|
}
|
||||||
|
}
|
||||||
65
scm-ui/public/locales/es/users.json
Normal file
65
scm-ui/public/locales/es/users.json
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
{
|
||||||
|
"user": {
|
||||||
|
"name": "Nombre de usuario",
|
||||||
|
"displayName": "Nombre a mostrar",
|
||||||
|
"mail": "Correo electrónico",
|
||||||
|
"password": "Contraseña",
|
||||||
|
"active": "Activo",
|
||||||
|
"inactive": "Inactivo",
|
||||||
|
"type": "Tipo",
|
||||||
|
"creationDate": "Fecha de creación",
|
||||||
|
"lastModified": "Última modificación"
|
||||||
|
},
|
||||||
|
"validation": {
|
||||||
|
"mail-invalid": "El correo electrónico es incorrecto",
|
||||||
|
"name-invalid": "El nombre es incorrecto",
|
||||||
|
"displayname-invalid": "El nombre a mostrar es incorrecto"
|
||||||
|
},
|
||||||
|
"help": {
|
||||||
|
"usernameHelpText": "Nombre único del usuario.",
|
||||||
|
"displayNameHelpText": "Nombre de usuario a mostrar.",
|
||||||
|
"mailHelpText": "Dirección de correo electrónico del usuario.",
|
||||||
|
"adminHelpText": "Un administrador es capaz de crear, modificar y borrar repositorios, grupos y usuarios.",
|
||||||
|
"activeHelpText": "Activar o desactivar el usuario."
|
||||||
|
},
|
||||||
|
"users": {
|
||||||
|
"title": "Usuarios",
|
||||||
|
"subtitle": "Crear, leer, actualizar y borrar usuarios",
|
||||||
|
"noUsers": "No se han encontrado usuarios.",
|
||||||
|
"createButton": "Crear usuario"
|
||||||
|
},
|
||||||
|
"singleUser": {
|
||||||
|
"errorTitle": "Error",
|
||||||
|
"errorSubtitle": "Error de usuario desconocido",
|
||||||
|
"menu": {
|
||||||
|
"navigationLabel": "Menú de usuario",
|
||||||
|
"informationNavLink": "Información",
|
||||||
|
"settingsNavLink": "Ajustes",
|
||||||
|
"generalNavLink": "General",
|
||||||
|
"setPasswordNavLink": "Contraseña",
|
||||||
|
"setPermissionsNavLink": "Permisos"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"createUser": {
|
||||||
|
"title": "Crear usuario",
|
||||||
|
"subtitle": "Crear un nuevo usuario"
|
||||||
|
},
|
||||||
|
"deleteUser": {
|
||||||
|
"subtitle": "Borrar usuario",
|
||||||
|
"button": "Borrar",
|
||||||
|
"confirmAlert": {
|
||||||
|
"title": "Borrar usuario",
|
||||||
|
"message": "¿Realmente desea borrar el usuario?",
|
||||||
|
"submit": "Sí",
|
||||||
|
"cancel": "No"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"singleUserPassword": {
|
||||||
|
"button": "Guardar",
|
||||||
|
"setPasswordSuccessful": "Contraseña guardada correctamente"
|
||||||
|
},
|
||||||
|
"userForm": {
|
||||||
|
"subtitle": "Editar usuario",
|
||||||
|
"button": "Guardar"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,12 +1,12 @@
|
|||||||
// @flow
|
// @flow
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { Button } from "@scm-manager/ui-components";
|
import { Button } from "@scm-manager/ui-components";
|
||||||
import type { PluginCollection } from "@scm-manager/ui-types";
|
import type { PendingPlugins } from "@scm-manager/ui-types";
|
||||||
import { translate } from "react-i18next";
|
import { translate } from "react-i18next";
|
||||||
import InstallPendingModal from "./InstallPendingModal";
|
import ExecutePendingModal from "./ExecutePendingModal";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
collection: PluginCollection,
|
pendingPlugins: PendingPlugins,
|
||||||
|
|
||||||
// context props
|
// context props
|
||||||
t: string => string
|
t: string => string
|
||||||
@@ -16,7 +16,7 @@ type State = {
|
|||||||
showModal: boolean
|
showModal: boolean
|
||||||
};
|
};
|
||||||
|
|
||||||
class InstallPendingAction extends React.Component<Props, State> {
|
class ExecutePendingAction extends React.Component<Props, State> {
|
||||||
constructor(props: Props) {
|
constructor(props: Props) {
|
||||||
super(props);
|
super(props);
|
||||||
this.state = {
|
this.state = {
|
||||||
@@ -38,11 +38,11 @@ class InstallPendingAction extends React.Component<Props, State> {
|
|||||||
|
|
||||||
renderModal = () => {
|
renderModal = () => {
|
||||||
const { showModal } = this.state;
|
const { showModal } = this.state;
|
||||||
const { collection } = this.props;
|
const { pendingPlugins } = this.props;
|
||||||
if (showModal) {
|
if (showModal) {
|
||||||
return (
|
return (
|
||||||
<InstallPendingModal
|
<ExecutePendingModal
|
||||||
collection={collection}
|
pendingPlugins={pendingPlugins}
|
||||||
onClose={this.closeModal}
|
onClose={this.closeModal}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
@@ -57,7 +57,7 @@ class InstallPendingAction extends React.Component<Props, State> {
|
|||||||
{this.renderModal()}
|
{this.renderModal()}
|
||||||
<Button
|
<Button
|
||||||
color="primary"
|
color="primary"
|
||||||
label={t("plugins.installPending")}
|
label={t("plugins.executePending")}
|
||||||
action={this.openModal}
|
action={this.openModal}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
@@ -65,4 +65,4 @@ class InstallPendingAction extends React.Component<Props, State> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default translate("admin")(InstallPendingAction);
|
export default translate("admin")(ExecutePendingAction);
|
||||||
185
scm-ui/src/admin/plugins/components/ExecutePendingModal.js
Normal file
185
scm-ui/src/admin/plugins/components/ExecutePendingModal.js
Normal file
@@ -0,0 +1,185 @@
|
|||||||
|
// @flow
|
||||||
|
import React from "react";
|
||||||
|
import {
|
||||||
|
apiClient,
|
||||||
|
Button,
|
||||||
|
ButtonGroup,
|
||||||
|
ErrorNotification,
|
||||||
|
Modal,
|
||||||
|
Notification
|
||||||
|
} from "@scm-manager/ui-components";
|
||||||
|
import type { PendingPlugins } from "@scm-manager/ui-types";
|
||||||
|
import { translate } from "react-i18next";
|
||||||
|
import waitForRestart from "./waitForRestart";
|
||||||
|
import SuccessNotification from "./SuccessNotification";
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
onClose: () => void,
|
||||||
|
pendingPlugins: PendingPlugins,
|
||||||
|
|
||||||
|
// context props
|
||||||
|
t: string => string
|
||||||
|
};
|
||||||
|
|
||||||
|
type State = {
|
||||||
|
loading: boolean,
|
||||||
|
success: boolean,
|
||||||
|
error?: Error
|
||||||
|
};
|
||||||
|
|
||||||
|
class ExecutePendingModal extends React.Component<Props, State> {
|
||||||
|
constructor(props: Props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
loading: false,
|
||||||
|
success: false
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
renderNotifications = () => {
|
||||||
|
const { t } = this.props;
|
||||||
|
const { error, success } = this.state;
|
||||||
|
if (error) {
|
||||||
|
return <ErrorNotification error={error} />;
|
||||||
|
} else if (success) {
|
||||||
|
return <SuccessNotification />;
|
||||||
|
} else {
|
||||||
|
return (
|
||||||
|
<Notification type="warning">
|
||||||
|
{t("plugins.modal.restartNotification")}
|
||||||
|
</Notification>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
executeAndRestart = () => {
|
||||||
|
const { pendingPlugins } = this.props;
|
||||||
|
this.setState({
|
||||||
|
loading: true
|
||||||
|
});
|
||||||
|
|
||||||
|
apiClient
|
||||||
|
.post(pendingPlugins._links.execute.href)
|
||||||
|
.then(waitForRestart)
|
||||||
|
.then(() => {
|
||||||
|
this.setState({
|
||||||
|
success: true,
|
||||||
|
loading: false,
|
||||||
|
error: undefined
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
this.setState({
|
||||||
|
success: false,
|
||||||
|
loading: false,
|
||||||
|
error: error
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
renderInstallQueue = () => {
|
||||||
|
const { pendingPlugins, t } = this.props;
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{pendingPlugins._embedded &&
|
||||||
|
pendingPlugins._embedded.new.length > 0 && (
|
||||||
|
<>
|
||||||
|
<strong>{t("plugins.modal.installQueue")}</strong>
|
||||||
|
<ul>
|
||||||
|
{pendingPlugins._embedded.new.map(plugin => (
|
||||||
|
<li key={plugin.name}>{plugin.name}</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
renderUpdateQueue = () => {
|
||||||
|
const { pendingPlugins, t } = this.props;
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{pendingPlugins._embedded &&
|
||||||
|
pendingPlugins._embedded.update.length > 0 && (
|
||||||
|
<>
|
||||||
|
<strong>{t("plugins.modal.updateQueue")}</strong>
|
||||||
|
<ul>
|
||||||
|
{pendingPlugins._embedded.update.map(plugin => (
|
||||||
|
<li key={plugin.name}>{plugin.name}</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
renderUninstallQueue = () => {
|
||||||
|
const { pendingPlugins, t } = this.props;
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{pendingPlugins._embedded &&
|
||||||
|
pendingPlugins._embedded.uninstall.length > 0 && (
|
||||||
|
<>
|
||||||
|
<strong>{t("plugins.modal.uninstallQueue")}</strong>
|
||||||
|
<ul>
|
||||||
|
{pendingPlugins._embedded.uninstall.map(plugin => (
|
||||||
|
<li key={plugin.name}>{plugin.name}</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
renderBody = () => {
|
||||||
|
const { t } = this.props;
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div className="media">
|
||||||
|
<div className="content">
|
||||||
|
<p>{t("plugins.modal.executePending")}</p>
|
||||||
|
{this.renderInstallQueue()}
|
||||||
|
{this.renderUpdateQueue()}
|
||||||
|
{this.renderUninstallQueue()}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="media">{this.renderNotifications()}</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
renderFooter = () => {
|
||||||
|
const { onClose, t } = this.props;
|
||||||
|
const { loading, error, success } = this.state;
|
||||||
|
return (
|
||||||
|
<ButtonGroup>
|
||||||
|
<Button
|
||||||
|
color="warning"
|
||||||
|
label={t("plugins.modal.executeAndRestart")}
|
||||||
|
loading={loading}
|
||||||
|
action={this.executeAndRestart}
|
||||||
|
disabled={error || success}
|
||||||
|
/>
|
||||||
|
<Button label={t("plugins.modal.abort")} action={onClose} />
|
||||||
|
</ButtonGroup>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { onClose, t } = this.props;
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
title={t("plugins.modal.executeAndRestart")}
|
||||||
|
closeFunction={onClose}
|
||||||
|
body={this.renderBody()}
|
||||||
|
footer={this.renderFooter()}
|
||||||
|
active={true}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default translate("admin")(ExecutePendingModal);
|
||||||
@@ -1,134 +0,0 @@
|
|||||||
// @flow
|
|
||||||
import React from "react";
|
|
||||||
import {
|
|
||||||
apiClient,
|
|
||||||
Button,
|
|
||||||
ButtonGroup,
|
|
||||||
ErrorNotification,
|
|
||||||
Modal,
|
|
||||||
Notification
|
|
||||||
} from "@scm-manager/ui-components";
|
|
||||||
import type { PluginCollection } from "@scm-manager/ui-types";
|
|
||||||
import { translate } from "react-i18next";
|
|
||||||
import waitForRestart from "./waitForRestart";
|
|
||||||
import InstallSuccessNotification from "./InstallSuccessNotification";
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
onClose: () => void,
|
|
||||||
collection: PluginCollection,
|
|
||||||
|
|
||||||
// context props
|
|
||||||
t: string => string
|
|
||||||
};
|
|
||||||
|
|
||||||
type State = {
|
|
||||||
loading: boolean,
|
|
||||||
success: boolean,
|
|
||||||
error?: Error
|
|
||||||
};
|
|
||||||
|
|
||||||
class InstallPendingModal extends React.Component<Props, State> {
|
|
||||||
constructor(props: Props) {
|
|
||||||
super(props);
|
|
||||||
this.state = {
|
|
||||||
loading: false,
|
|
||||||
success: false
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
renderNotifications = () => {
|
|
||||||
const { t } = this.props;
|
|
||||||
const { error, success } = this.state;
|
|
||||||
if (error) {
|
|
||||||
return <ErrorNotification error={error} />;
|
|
||||||
} else if (success) {
|
|
||||||
return <InstallSuccessNotification />;
|
|
||||||
} else {
|
|
||||||
return (
|
|
||||||
<Notification type="warning">
|
|
||||||
{t("plugins.modal.restartNotification")}
|
|
||||||
</Notification>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
installAndRestart = () => {
|
|
||||||
const { collection } = this.props;
|
|
||||||
this.setState({
|
|
||||||
loading: true
|
|
||||||
});
|
|
||||||
|
|
||||||
apiClient
|
|
||||||
.post(collection._links.installPending.href)
|
|
||||||
.then(waitForRestart)
|
|
||||||
.then(() => {
|
|
||||||
this.setState({
|
|
||||||
success: true,
|
|
||||||
loading: false,
|
|
||||||
error: undefined
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.catch(error => {
|
|
||||||
this.setState({
|
|
||||||
success: false,
|
|
||||||
loading: false,
|
|
||||||
error: error
|
|
||||||
});
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
renderBody = () => {
|
|
||||||
const { collection, t } = this.props;
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<div className="media">
|
|
||||||
<div className="content">
|
|
||||||
<p>{t("plugins.modal.installPending")}</p>
|
|
||||||
<ul>
|
|
||||||
{collection._embedded.plugins
|
|
||||||
.filter(plugin => plugin.pending)
|
|
||||||
.map(plugin => (
|
|
||||||
<li key={plugin.name} className="has-text-weight-bold">
|
|
||||||
{plugin.name}
|
|
||||||
</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className="media">{this.renderNotifications()}</div>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
renderFooter = () => {
|
|
||||||
const { onClose, t } = this.props;
|
|
||||||
const { loading, error, success } = this.state;
|
|
||||||
return (
|
|
||||||
<ButtonGroup>
|
|
||||||
<Button
|
|
||||||
color="warning"
|
|
||||||
label={t("plugins.modal.installAndRestart")}
|
|
||||||
loading={loading}
|
|
||||||
action={this.installAndRestart}
|
|
||||||
disabled={error || success}
|
|
||||||
/>
|
|
||||||
<Button label={t("plugins.modal.abort")} action={onClose} />
|
|
||||||
</ButtonGroup>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
render() {
|
|
||||||
const { onClose, t } = this.props;
|
|
||||||
return (
|
|
||||||
<Modal
|
|
||||||
title={t("plugins.modal.installAndRestart")}
|
|
||||||
closeFunction={onClose}
|
|
||||||
body={this.renderBody()}
|
|
||||||
footer={this.renderFooter()}
|
|
||||||
active={true}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export default translate("admin")(InstallPendingModal);
|
|
||||||
@@ -4,8 +4,14 @@ import injectSheet from "react-jss";
|
|||||||
import type { Plugin } from "@scm-manager/ui-types";
|
import type { Plugin } from "@scm-manager/ui-types";
|
||||||
import { CardColumn } from "@scm-manager/ui-components";
|
import { CardColumn } from "@scm-manager/ui-components";
|
||||||
import PluginAvatar from "./PluginAvatar";
|
import PluginAvatar from "./PluginAvatar";
|
||||||
import PluginModal from "./PluginModal";
|
|
||||||
import classNames from "classnames";
|
import classNames from "classnames";
|
||||||
|
import PluginModal from "./PluginModal";
|
||||||
|
|
||||||
|
export const PluginAction = {
|
||||||
|
INSTALL: "install",
|
||||||
|
UPDATE: "update",
|
||||||
|
UNINSTALL: "uninstall"
|
||||||
|
};
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
plugin: Plugin,
|
plugin: Plugin,
|
||||||
@@ -16,18 +22,37 @@ type Props = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
type State = {
|
type State = {
|
||||||
showModal: boolean
|
showInstallModal: boolean,
|
||||||
|
showUpdateModal: boolean,
|
||||||
|
showUninstallModal: boolean
|
||||||
};
|
};
|
||||||
|
|
||||||
const styles = {
|
const styles = {
|
||||||
link: {
|
link: {
|
||||||
cursor: "pointer",
|
cursor: "pointer",
|
||||||
pointerEvents: "all"
|
pointerEvents: "all",
|
||||||
|
padding: "0.5rem",
|
||||||
|
border: "solid 1px var(--dark-25)",
|
||||||
|
borderRadius: "4px",
|
||||||
|
"&:hover": {
|
||||||
|
borderColor: "var(--dark-50)"
|
||||||
|
}
|
||||||
},
|
},
|
||||||
spinner: {
|
topRight: {
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
right: 0,
|
right: 0,
|
||||||
top: 0
|
top: 0
|
||||||
|
},
|
||||||
|
layout: {
|
||||||
|
"& .level": {
|
||||||
|
paddingBottom: "0.5rem"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
actionbar: {
|
||||||
|
display: "flex",
|
||||||
|
"& span + span": {
|
||||||
|
marginLeft: "0.5rem"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -36,7 +61,9 @@ class PluginEntry extends React.Component<Props, State> {
|
|||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
showModal: false
|
showInstallModal: false,
|
||||||
|
showUpdateModal: false,
|
||||||
|
showUninstallModal: false
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -44,10 +71,9 @@ class PluginEntry extends React.Component<Props, State> {
|
|||||||
return <PluginAvatar plugin={plugin} />;
|
return <PluginAvatar plugin={plugin} />;
|
||||||
};
|
};
|
||||||
|
|
||||||
toggleModal = () => {
|
toggleModal = (showModal: string) => {
|
||||||
this.setState(prevState => ({
|
const oldValue = this.state[showModal];
|
||||||
showModal: !prevState.showModal
|
this.setState({ [showModal]: !oldValue });
|
||||||
}));
|
|
||||||
};
|
};
|
||||||
|
|
||||||
createFooterRight = (plugin: Plugin) => {
|
createFooterRight = (plugin: Plugin) => {
|
||||||
@@ -59,56 +85,123 @@ class PluginEntry extends React.Component<Props, State> {
|
|||||||
return plugin._links && plugin._links.install && plugin._links.install.href;
|
return plugin._links && plugin._links.install && plugin._links.install.href;
|
||||||
};
|
};
|
||||||
|
|
||||||
createFooterLeft = () => {
|
isUpdatable = () => {
|
||||||
const { classes } = this.props;
|
const { plugin } = this.props;
|
||||||
if (this.isInstallable()) {
|
return plugin._links && plugin._links.update && plugin._links.update.href;
|
||||||
|
};
|
||||||
|
|
||||||
|
isUninstallable = () => {
|
||||||
|
const { plugin } = this.props;
|
||||||
return (
|
return (
|
||||||
|
plugin._links && plugin._links.uninstall && plugin._links.uninstall.href
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
createActionbar = () => {
|
||||||
|
const { classes } = this.props;
|
||||||
|
return (
|
||||||
|
<div className={classNames(classes.actionbar, classes.topRight)}>
|
||||||
|
{this.isInstallable() && (
|
||||||
<span
|
<span
|
||||||
className={classNames(classes.link, "level-item")}
|
className={classNames(classes.link, "level-item")}
|
||||||
onClick={this.toggleModal}
|
onClick={() => this.toggleModal("showInstallModal")}
|
||||||
>
|
>
|
||||||
<i className="fas fa-download has-text-info" />
|
<i className="fas fa-download has-text-info" />
|
||||||
</span>
|
</span>
|
||||||
|
)}
|
||||||
|
{this.isUninstallable() && (
|
||||||
|
<span
|
||||||
|
className={classNames(classes.link, "level-item")}
|
||||||
|
onClick={() => this.toggleModal("showUninstallModal")}
|
||||||
|
>
|
||||||
|
<i className="fas fa-trash has-text-info" />
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
{this.isUpdatable() && (
|
||||||
|
<span
|
||||||
|
className={classNames(classes.link, "level-item")}
|
||||||
|
onClick={() => this.toggleModal("showUpdateModal")}
|
||||||
|
>
|
||||||
|
<i className="fas fa-sync-alt has-text-info" />
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
renderModal = () => {
|
||||||
|
const { plugin, refresh } = this.props;
|
||||||
|
if (this.state.showInstallModal && this.isInstallable()) {
|
||||||
|
return (
|
||||||
|
<PluginModal
|
||||||
|
plugin={plugin}
|
||||||
|
pluginAction={PluginAction.INSTALL}
|
||||||
|
refresh={refresh}
|
||||||
|
onClose={() => this.toggleModal("showInstallModal")}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} else if (this.state.showUpdateModal && this.isUpdatable()) {
|
||||||
|
return (
|
||||||
|
<PluginModal
|
||||||
|
plugin={plugin}
|
||||||
|
pluginAction={PluginAction.UPDATE}
|
||||||
|
refresh={refresh}
|
||||||
|
onClose={() => this.toggleModal("showUpdateModal")}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} else if (this.state.showUninstallModal && this.isUninstallable()) {
|
||||||
|
return (
|
||||||
|
<PluginModal
|
||||||
|
plugin={plugin}
|
||||||
|
pluginAction={PluginAction.UNINSTALL}
|
||||||
|
refresh={refresh}
|
||||||
|
onClose={() => this.toggleModal("showUninstallModal")}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
createPendingSpinner = () => {
|
createPendingSpinner = () => {
|
||||||
const { plugin, classes } = this.props;
|
const { plugin, classes } = this.props;
|
||||||
if (plugin.pending) {
|
|
||||||
return (
|
return (
|
||||||
<span className={classes.spinner}>
|
<span className={classes.topRight}>
|
||||||
<i className="fas fa-spinner fa-spin has-text-info" />
|
<i
|
||||||
|
className={classNames(
|
||||||
|
"fas fa-spinner fa-lg fa-spin",
|
||||||
|
plugin.markedForUninstall ? "has-text-danger" : "has-text-info"
|
||||||
|
)}
|
||||||
|
/>
|
||||||
</span>
|
</span>
|
||||||
);
|
);
|
||||||
}
|
|
||||||
return null;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { plugin, refresh } = this.props;
|
const { plugin, classes } = this.props;
|
||||||
const { showModal } = this.state;
|
|
||||||
const avatar = this.createAvatar(plugin);
|
const avatar = this.createAvatar(plugin);
|
||||||
const footerLeft = this.createFooterLeft();
|
const actionbar = this.createActionbar();
|
||||||
const footerRight = this.createFooterRight(plugin);
|
const footerRight = this.createFooterRight(plugin);
|
||||||
|
|
||||||
const modal = showModal ? (
|
const modal = this.renderModal();
|
||||||
<PluginModal
|
|
||||||
plugin={plugin}
|
|
||||||
refresh={refresh}
|
|
||||||
onClose={this.toggleModal}
|
|
||||||
/>
|
|
||||||
) : null;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<CardColumn
|
<CardColumn
|
||||||
action={this.isInstallable() ? this.toggleModal : null}
|
className={classes.layout}
|
||||||
|
action={
|
||||||
|
this.isInstallable()
|
||||||
|
? () => this.toggleModal("showInstallModal")
|
||||||
|
: null
|
||||||
|
}
|
||||||
avatar={avatar}
|
avatar={avatar}
|
||||||
title={plugin.displayName ? plugin.displayName : plugin.name}
|
title={plugin.displayName ? plugin.displayName : plugin.name}
|
||||||
description={plugin.description}
|
description={plugin.description}
|
||||||
contentRight={this.createPendingSpinner()}
|
contentRight={
|
||||||
footerLeft={footerLeft}
|
plugin.pending || plugin.markedForUninstall
|
||||||
|
? this.createPendingSpinner()
|
||||||
|
: actionbar
|
||||||
|
}
|
||||||
footerRight={footerRight}
|
footerRight={footerRight}
|
||||||
/>
|
/>
|
||||||
{modal}
|
{modal}
|
||||||
|
|||||||
@@ -15,10 +15,12 @@ import {
|
|||||||
} from "@scm-manager/ui-components";
|
} from "@scm-manager/ui-components";
|
||||||
import classNames from "classnames";
|
import classNames from "classnames";
|
||||||
import waitForRestart from "./waitForRestart";
|
import waitForRestart from "./waitForRestart";
|
||||||
import InstallSuccessNotification from "./InstallSuccessNotification";
|
import SuccessNotification from "./SuccessNotification";
|
||||||
|
import { PluginAction } from "./PluginEntry";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
plugin: Plugin,
|
plugin: Plugin,
|
||||||
|
pluginAction: string,
|
||||||
refresh: () => void,
|
refresh: () => void,
|
||||||
onClose: () => void,
|
onClose: () => void,
|
||||||
|
|
||||||
@@ -37,9 +39,14 @@ type State = {
|
|||||||
const styles = {
|
const styles = {
|
||||||
userLabelAlignment: {
|
userLabelAlignment: {
|
||||||
textAlign: "left",
|
textAlign: "left",
|
||||||
marginRight: 0,
|
marginRight: 0
|
||||||
|
},
|
||||||
|
userLabelMarginSmall: {
|
||||||
minWidth: "5.5em"
|
minWidth: "5.5em"
|
||||||
},
|
},
|
||||||
|
userLabelMarginLarge: {
|
||||||
|
minWidth: "10em"
|
||||||
|
},
|
||||||
userFieldFlex: {
|
userFieldFlex: {
|
||||||
flexGrow: 4
|
flexGrow: 4
|
||||||
}
|
}
|
||||||
@@ -55,7 +62,7 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
onInstallSuccess = () => {
|
onSuccess = () => {
|
||||||
const { restart } = this.state;
|
const { restart } = this.state;
|
||||||
const { refresh, onClose } = this.props;
|
const { refresh, onClose } = this.props;
|
||||||
|
|
||||||
@@ -87,16 +94,30 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
install = (e: Event) => {
|
createPluginActionLink = () => {
|
||||||
|
const { plugin, pluginAction } = this.props;
|
||||||
const { restart } = this.state;
|
const { restart } = this.state;
|
||||||
const { plugin } = this.props;
|
|
||||||
|
let pluginActionLink = "";
|
||||||
|
|
||||||
|
if (pluginAction === PluginAction.INSTALL) {
|
||||||
|
pluginActionLink = plugin._links.install.href;
|
||||||
|
} else if (pluginAction === PluginAction.UPDATE) {
|
||||||
|
pluginActionLink = plugin._links.update.href;
|
||||||
|
} else if (pluginAction === PluginAction.UNINSTALL) {
|
||||||
|
pluginActionLink = plugin._links.uninstall.href;
|
||||||
|
}
|
||||||
|
return pluginActionLink + "?restart=" + restart.toString();
|
||||||
|
};
|
||||||
|
|
||||||
|
handlePluginAction = (e: Event) => {
|
||||||
this.setState({
|
this.setState({
|
||||||
loading: true
|
loading: true
|
||||||
});
|
});
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
apiClient
|
apiClient
|
||||||
.post(plugin._links.install.href + "?restart=" + restart.toString())
|
.post(this.createPluginActionLink())
|
||||||
.then(this.onInstallSuccess)
|
.then(this.onSuccess)
|
||||||
.catch(error => {
|
.catch(error => {
|
||||||
this.setState({
|
this.setState({
|
||||||
loading: false,
|
loading: false,
|
||||||
@@ -106,21 +127,21 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
footer = () => {
|
footer = () => {
|
||||||
const { onClose, t } = this.props;
|
const { pluginAction, onClose, t } = this.props;
|
||||||
const { loading, error, restart, success } = this.state;
|
const { loading, error, restart, success } = this.state;
|
||||||
|
|
||||||
let color = "primary";
|
let color = pluginAction === PluginAction.UNINSTALL ? "warning" : "primary";
|
||||||
let label = "plugins.modal.install";
|
let label = `plugins.modal.${pluginAction}`;
|
||||||
if (restart) {
|
if (restart) {
|
||||||
color = "warning";
|
color = "warning";
|
||||||
label = "plugins.modal.installAndRestart";
|
label = `plugins.modal.${pluginAction}AndRestart`;
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<ButtonGroup>
|
<ButtonGroup>
|
||||||
<Button
|
<Button
|
||||||
label={t(label)}
|
label={t(label)}
|
||||||
color={color}
|
color={color}
|
||||||
action={this.install}
|
action={this.handlePluginAction}
|
||||||
loading={loading}
|
loading={loading}
|
||||||
disabled={!!error || success}
|
disabled={!!error || success}
|
||||||
/>
|
/>
|
||||||
@@ -162,7 +183,7 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
} else if (success) {
|
} else if (success) {
|
||||||
return (
|
return (
|
||||||
<div className="media">
|
<div className="media">
|
||||||
<InstallSuccessNotification />
|
<SuccessNotification />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
} else if (restart) {
|
} else if (restart) {
|
||||||
@@ -185,7 +206,7 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { restart } = this.state;
|
const { restart } = this.state;
|
||||||
const { plugin, onClose, classes, t } = this.props;
|
const { plugin, pluginAction, onClose, classes, t } = this.props;
|
||||||
|
|
||||||
const body = (
|
const body = (
|
||||||
<>
|
<>
|
||||||
@@ -200,6 +221,9 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
<div
|
<div
|
||||||
className={classNames(
|
className={classNames(
|
||||||
classes.userLabelAlignment,
|
classes.userLabelAlignment,
|
||||||
|
pluginAction === PluginAction.INSTALL
|
||||||
|
? classes.userLabelMarginSmall
|
||||||
|
: classes.userLabelMarginLarge,
|
||||||
"field-label is-inline-flex"
|
"field-label is-inline-flex"
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
@@ -214,10 +238,12 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
{plugin.author}
|
{plugin.author}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{pluginAction === PluginAction.INSTALL && (
|
||||||
<div className="field is-horizontal">
|
<div className="field is-horizontal">
|
||||||
<div
|
<div
|
||||||
className={classNames(
|
className={classNames(
|
||||||
classes.userLabelAlignment,
|
classes.userLabelAlignment,
|
||||||
|
classes.userLabelMarginSmall,
|
||||||
"field-label is-inline-flex"
|
"field-label is-inline-flex"
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
@@ -232,7 +258,50 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
{plugin.version}
|
{plugin.version}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
)}
|
||||||
|
{(pluginAction === PluginAction.UPDATE ||
|
||||||
|
pluginAction === PluginAction.UNINSTALL) && (
|
||||||
|
<div className="field is-horizontal">
|
||||||
|
<div
|
||||||
|
className={classNames(
|
||||||
|
classes.userLabelAlignment,
|
||||||
|
classes.userLabelMarginLarge,
|
||||||
|
"field-label is-inline-flex"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{t("plugins.modal.currentVersion")}:
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className={classNames(
|
||||||
|
classes.userFieldFlex,
|
||||||
|
"field-body is-inline-flex"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{plugin.version}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{pluginAction === PluginAction.UPDATE && (
|
||||||
|
<div className="field is-horizontal">
|
||||||
|
<div
|
||||||
|
className={classNames(
|
||||||
|
classes.userLabelAlignment,
|
||||||
|
classes.userLabelMarginLarge,
|
||||||
|
"field-label is-inline-flex"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{t("plugins.modal.newVersion")}:
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className={classNames(
|
||||||
|
classes.userFieldFlex,
|
||||||
|
"field-body is-inline-flex"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{plugin.newVersion}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
{this.renderDependencies()}
|
{this.renderDependencies()}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -252,7 +321,7 @@ class PluginModal extends React.Component<Props, State> {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
title={t("plugins.modal.title", {
|
title={t(`plugins.modal.title.${pluginAction}`, {
|
||||||
name: plugin.displayName ? plugin.displayName : plugin.name
|
name: plugin.displayName ? plugin.displayName : plugin.name
|
||||||
})}
|
})}
|
||||||
closeFunction={() => onClose()}
|
closeFunction={() => onClose()}
|
||||||
|
|||||||
@@ -3,28 +3,31 @@ import * as React from "react";
|
|||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { translate } from "react-i18next";
|
import { translate } from "react-i18next";
|
||||||
import { compose } from "redux";
|
import { compose } from "redux";
|
||||||
import type { PluginCollection } from "@scm-manager/ui-types";
|
import type { PendingPlugins, PluginCollection } from "@scm-manager/ui-types";
|
||||||
import {
|
import {
|
||||||
|
ErrorNotification,
|
||||||
Loading,
|
Loading,
|
||||||
Title,
|
|
||||||
Subtitle,
|
|
||||||
Notification,
|
Notification,
|
||||||
ErrorNotification
|
Subtitle,
|
||||||
|
Title
|
||||||
} from "@scm-manager/ui-components";
|
} from "@scm-manager/ui-components";
|
||||||
import {
|
import {
|
||||||
|
fetchPendingPlugins,
|
||||||
fetchPluginsByLink,
|
fetchPluginsByLink,
|
||||||
getFetchPluginsFailure,
|
getFetchPluginsFailure,
|
||||||
|
getPendingPlugins,
|
||||||
getPluginCollection,
|
getPluginCollection,
|
||||||
isFetchPluginsPending
|
isFetchPluginsPending
|
||||||
} from "../modules/plugins";
|
} from "../modules/plugins";
|
||||||
import PluginsList from "../components/PluginList";
|
import PluginsList from "../components/PluginList";
|
||||||
import {
|
import {
|
||||||
getAvailablePluginsLink,
|
getAvailablePluginsLink,
|
||||||
getInstalledPluginsLink
|
getInstalledPluginsLink,
|
||||||
|
getPendingPluginsLink
|
||||||
} from "../../../modules/indexResource";
|
} from "../../../modules/indexResource";
|
||||||
import PluginTopActions from "../components/PluginTopActions";
|
import PluginTopActions from "../components/PluginTopActions";
|
||||||
import PluginBottomActions from "../components/PluginBottomActions";
|
import PluginBottomActions from "../components/PluginBottomActions";
|
||||||
import InstallPendingAction from "../components/InstallPendingAction";
|
import ExecutePendingAction from "../components/ExecutePendingAction";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
loading: boolean,
|
loading: boolean,
|
||||||
@@ -34,12 +37,15 @@ type Props = {
|
|||||||
installed: boolean,
|
installed: boolean,
|
||||||
availablePluginsLink: string,
|
availablePluginsLink: string,
|
||||||
installedPluginsLink: string,
|
installedPluginsLink: string,
|
||||||
|
pendingPluginsLink: string,
|
||||||
|
pendingPlugins: PendingPlugins,
|
||||||
|
|
||||||
// context objects
|
// context objects
|
||||||
t: string => string,
|
t: string => string,
|
||||||
|
|
||||||
// dispatched functions
|
// dispatched functions
|
||||||
fetchPluginsByLink: (link: string) => void
|
fetchPluginsByLink: (link: string) => void,
|
||||||
|
fetchPendingPlugins: (link: string) => void
|
||||||
};
|
};
|
||||||
|
|
||||||
class PluginsOverview extends React.Component<Props> {
|
class PluginsOverview extends React.Component<Props> {
|
||||||
@@ -48,15 +54,16 @@ class PluginsOverview extends React.Component<Props> {
|
|||||||
installed,
|
installed,
|
||||||
fetchPluginsByLink,
|
fetchPluginsByLink,
|
||||||
availablePluginsLink,
|
availablePluginsLink,
|
||||||
installedPluginsLink
|
installedPluginsLink,
|
||||||
|
pendingPluginsLink,
|
||||||
|
fetchPendingPlugins
|
||||||
} = this.props;
|
} = this.props;
|
||||||
fetchPluginsByLink(installed ? installedPluginsLink : availablePluginsLink);
|
fetchPluginsByLink(installed ? installedPluginsLink : availablePluginsLink);
|
||||||
|
fetchPendingPlugins(pendingPluginsLink);
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidUpdate(prevProps) {
|
componentDidUpdate(prevProps) {
|
||||||
const {
|
const { installed } = this.props;
|
||||||
installed,
|
|
||||||
} = this.props;
|
|
||||||
if (prevProps.installed !== installed) {
|
if (prevProps.installed !== installed) {
|
||||||
this.fetchPlugins();
|
this.fetchPlugins();
|
||||||
}
|
}
|
||||||
@@ -67,11 +74,12 @@ class PluginsOverview extends React.Component<Props> {
|
|||||||
installed,
|
installed,
|
||||||
fetchPluginsByLink,
|
fetchPluginsByLink,
|
||||||
availablePluginsLink,
|
availablePluginsLink,
|
||||||
installedPluginsLink
|
installedPluginsLink,
|
||||||
|
pendingPluginsLink,
|
||||||
|
fetchPendingPlugins
|
||||||
} = this.props;
|
} = this.props;
|
||||||
fetchPluginsByLink(
|
fetchPluginsByLink(installed ? installedPluginsLink : availablePluginsLink);
|
||||||
installed ? installedPluginsLink : availablePluginsLink
|
fetchPendingPlugins(pendingPluginsLink);
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
renderHeader = (actions: React.Node) => {
|
renderHeader = (actions: React.Node) => {
|
||||||
@@ -101,9 +109,13 @@ class PluginsOverview extends React.Component<Props> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
createActions = () => {
|
createActions = () => {
|
||||||
const { collection } = this.props;
|
const { pendingPlugins } = this.props;
|
||||||
if (collection._links.installPending) {
|
if (
|
||||||
return <InstallPendingAction collection={collection} />;
|
pendingPlugins &&
|
||||||
|
pendingPlugins._links &&
|
||||||
|
pendingPlugins._links.execute
|
||||||
|
) {
|
||||||
|
return <ExecutePendingAction pendingPlugins={pendingPlugins} />;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
};
|
};
|
||||||
@@ -134,7 +146,12 @@ class PluginsOverview extends React.Component<Props> {
|
|||||||
const { collection, t } = this.props;
|
const { collection, t } = this.props;
|
||||||
|
|
||||||
if (collection._embedded && collection._embedded.plugins.length > 0) {
|
if (collection._embedded && collection._embedded.plugins.length > 0) {
|
||||||
return <PluginsList plugins={collection._embedded.plugins} refresh={this.fetchPlugins} />;
|
return (
|
||||||
|
<PluginsList
|
||||||
|
plugins={collection._embedded.plugins}
|
||||||
|
refresh={this.fetchPlugins}
|
||||||
|
/>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
return <Notification type="info">{t("plugins.noPlugins")}</Notification>;
|
return <Notification type="info">{t("plugins.noPlugins")}</Notification>;
|
||||||
}
|
}
|
||||||
@@ -146,13 +163,17 @@ const mapStateToProps = state => {
|
|||||||
const error = getFetchPluginsFailure(state);
|
const error = getFetchPluginsFailure(state);
|
||||||
const availablePluginsLink = getAvailablePluginsLink(state);
|
const availablePluginsLink = getAvailablePluginsLink(state);
|
||||||
const installedPluginsLink = getInstalledPluginsLink(state);
|
const installedPluginsLink = getInstalledPluginsLink(state);
|
||||||
|
const pendingPluginsLink = getPendingPluginsLink(state);
|
||||||
|
const pendingPlugins = getPendingPlugins(state);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
collection,
|
collection,
|
||||||
loading,
|
loading,
|
||||||
error,
|
error,
|
||||||
availablePluginsLink,
|
availablePluginsLink,
|
||||||
installedPluginsLink
|
installedPluginsLink,
|
||||||
|
pendingPluginsLink,
|
||||||
|
pendingPlugins
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -160,6 +181,9 @@ const mapDispatchToProps = dispatch => {
|
|||||||
return {
|
return {
|
||||||
fetchPluginsByLink: (link: string) => {
|
fetchPluginsByLink: (link: string) => {
|
||||||
dispatch(fetchPluginsByLink(link));
|
dispatch(fetchPluginsByLink(link));
|
||||||
|
},
|
||||||
|
fetchPendingPlugins: (link: string) => {
|
||||||
|
dispatch(fetchPendingPlugins(link));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -15,6 +15,17 @@ export const FETCH_PLUGIN_PENDING = `${FETCH_PLUGIN}_${types.PENDING_SUFFIX}`;
|
|||||||
export const FETCH_PLUGIN_SUCCESS = `${FETCH_PLUGIN}_${types.SUCCESS_SUFFIX}`;
|
export const FETCH_PLUGIN_SUCCESS = `${FETCH_PLUGIN}_${types.SUCCESS_SUFFIX}`;
|
||||||
export const FETCH_PLUGIN_FAILURE = `${FETCH_PLUGIN}_${types.FAILURE_SUFFIX}`;
|
export const FETCH_PLUGIN_FAILURE = `${FETCH_PLUGIN}_${types.FAILURE_SUFFIX}`;
|
||||||
|
|
||||||
|
export const FETCH_PENDING_PLUGINS = "scm/plugins/FETCH_PENDING_PLUGINS";
|
||||||
|
export const FETCH_PENDING_PLUGINS_PENDING = `${FETCH_PENDING_PLUGINS}_${
|
||||||
|
types.PENDING_SUFFIX
|
||||||
|
}`;
|
||||||
|
export const FETCH_PENDING_PLUGINS_SUCCESS = `${FETCH_PENDING_PLUGINS}_${
|
||||||
|
types.SUCCESS_SUFFIX
|
||||||
|
}`;
|
||||||
|
export const FETCH_PENDING_PLUGINS_FAILURE = `${FETCH_PENDING_PLUGINS}_${
|
||||||
|
types.FAILURE_SUFFIX
|
||||||
|
}`;
|
||||||
|
|
||||||
// fetch plugins
|
// fetch plugins
|
||||||
export function fetchPluginsByLink(link: string) {
|
export function fetchPluginsByLink(link: string) {
|
||||||
return function(dispatch: any) {
|
return function(dispatch: any) {
|
||||||
@@ -105,8 +116,44 @@ export function fetchPluginFailure(name: string, error: Error): Action {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// fetch pending plugins
|
||||||
|
export function fetchPendingPlugins(link: string) {
|
||||||
|
return function(dispatch: any) {
|
||||||
|
dispatch(fetchPendingPluginsPending());
|
||||||
|
return apiClient
|
||||||
|
.get(link)
|
||||||
|
.then(response => response.json())
|
||||||
|
.then(PendingPlugins => {
|
||||||
|
dispatch(fetchPendingPluginsSuccess(PendingPlugins));
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
dispatch(fetchPendingPluginsFailure(err));
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function fetchPendingPluginsPending(): Action {
|
||||||
|
return {
|
||||||
|
type: FETCH_PENDING_PLUGINS_PENDING
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function fetchPendingPluginsSuccess(PendingPlugins: {}): Action {
|
||||||
|
return {
|
||||||
|
type: FETCH_PENDING_PLUGINS_SUCCESS,
|
||||||
|
payload: PendingPlugins
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function fetchPendingPluginsFailure(err: Error): Action {
|
||||||
|
return {
|
||||||
|
type: FETCH_PENDING_PLUGINS_FAILURE,
|
||||||
|
payload: err
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
// reducer
|
// reducer
|
||||||
function normalizeByName(pluginCollection: PluginCollection) {
|
function normalizeByName(state: Object, pluginCollection: PluginCollection) {
|
||||||
const names = [];
|
const names = [];
|
||||||
const byNames = {};
|
const byNames = {};
|
||||||
for (const plugin of pluginCollection._embedded.plugins) {
|
for (const plugin of pluginCollection._embedded.plugins) {
|
||||||
@@ -114,6 +161,7 @@ function normalizeByName(pluginCollection: PluginCollection) {
|
|||||||
byNames[plugin.name] = plugin;
|
byNames[plugin.name] = plugin;
|
||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
|
...state,
|
||||||
list: {
|
list: {
|
||||||
...pluginCollection,
|
...pluginCollection,
|
||||||
_embedded: {
|
_embedded: {
|
||||||
@@ -144,9 +192,11 @@ export default function reducer(
|
|||||||
|
|
||||||
switch (action.type) {
|
switch (action.type) {
|
||||||
case FETCH_PLUGINS_SUCCESS:
|
case FETCH_PLUGINS_SUCCESS:
|
||||||
return normalizeByName(action.payload);
|
return normalizeByName(state, action.payload);
|
||||||
case FETCH_PLUGIN_SUCCESS:
|
case FETCH_PLUGIN_SUCCESS:
|
||||||
return reducerByNames(state, action.payload);
|
return reducerByNames(state, action.payload);
|
||||||
|
case FETCH_PENDING_PLUGINS_SUCCESS:
|
||||||
|
return { ...state, pending: action.payload };
|
||||||
default:
|
default:
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
@@ -189,3 +239,17 @@ export function isFetchPluginPending(state: Object, name: string) {
|
|||||||
export function getFetchPluginFailure(state: Object, name: string) {
|
export function getFetchPluginFailure(state: Object, name: string) {
|
||||||
return getFailure(state, FETCH_PLUGIN, name);
|
return getFailure(state, FETCH_PLUGIN, name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getPendingPlugins(state: Object) {
|
||||||
|
if (state.plugins && state.plugins.pending) {
|
||||||
|
return state.plugins.pending;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isFetchPendingPluginsPending(state: Object) {
|
||||||
|
return isPending(state, FETCH_PENDING_PLUGINS);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getFetchPendingPluginsFailure(state: Object) {
|
||||||
|
return getFailure(state, FETCH_PENDING_PLUGINS);
|
||||||
|
}
|
||||||
|
|||||||
@@ -124,6 +124,10 @@ export function getInstalledPluginsLink(state: Object) {
|
|||||||
return getLink(state, "installedPlugins");
|
return getLink(state, "installedPlugins");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getPendingPluginsLink(state: Object) {
|
||||||
|
return getLink(state, "pendingPlugins");
|
||||||
|
}
|
||||||
|
|
||||||
export function getMeLink(state: Object) {
|
export function getMeLink(state: Object) {
|
||||||
return getLink(state, "me");
|
return getLink(state, "me");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,8 +43,29 @@ type Props = {
|
|||||||
class ChangesetsRoot extends React.Component<Props> {
|
class ChangesetsRoot extends React.Component<Props> {
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this.props.fetchBranches(this.props.repository);
|
this.props.fetchBranches(this.props.repository);
|
||||||
|
this.redirectToDefaultBranch();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
redirectToDefaultBranch = () => {
|
||||||
|
if (this.shouldRedirectToDefaultBranch()) {
|
||||||
|
const defaultBranches = this.props.branches.filter(
|
||||||
|
b => b.defaultBranch === true
|
||||||
|
);
|
||||||
|
if (defaultBranches.length > 0) {
|
||||||
|
this.branchSelected(defaultBranches[0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
shouldRedirectToDefaultBranch = () => {
|
||||||
|
return (
|
||||||
|
this.props.branches &&
|
||||||
|
this.props.branches.length > 0 &&
|
||||||
|
this.props.selected !==
|
||||||
|
this.props.branches.filter(b => b.defaultBranch === true)[0]
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
stripEndingSlash = (url: string) => {
|
stripEndingSlash = (url: string) => {
|
||||||
if (url.endsWith("/")) {
|
if (url.endsWith("/")) {
|
||||||
return url.substring(0, url.length - 1);
|
return url.substring(0, url.length - 1);
|
||||||
|
|||||||
@@ -106,7 +106,6 @@ class Content extends React.Component<Props, State> {
|
|||||||
</div>
|
</div>
|
||||||
<div className="buttons is-grouped">
|
<div className="buttons is-grouped">
|
||||||
<div className={classes.marginInHeader}>{selector}</div>
|
<div className={classes.marginInHeader}>{selector}</div>
|
||||||
<ButtonGroup>
|
|
||||||
<ExtensionPoint
|
<ExtensionPoint
|
||||||
name="repos.sources.content.actionbar"
|
name="repos.sources.content.actionbar"
|
||||||
props={{
|
props={{
|
||||||
@@ -116,7 +115,6 @@ class Content extends React.Component<Props, State> {
|
|||||||
}}
|
}}
|
||||||
renderAll={true}
|
renderAll={true}
|
||||||
/>
|
/>
|
||||||
</ButtonGroup>
|
|
||||||
</div>
|
</div>
|
||||||
</article>
|
</article>
|
||||||
</span>
|
</span>
|
||||||
|
|||||||
@@ -80,20 +80,17 @@ class Sources extends React.Component<Props, State> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
redirectToDefaultBranch = () => {
|
redirectToDefaultBranch = () => {
|
||||||
const { branches, baseUrl } = this.props;
|
const { branches } = this.props;
|
||||||
if (this.shouldRedirect()) {
|
if (this.shouldRedirectToDefaultBranch()) {
|
||||||
const defaultBranches = branches.filter(b => b.defaultBranch);
|
const defaultBranches = branches.filter(b => b.defaultBranch);
|
||||||
|
|
||||||
if (defaultBranches.length > 0) {
|
if (defaultBranches.length > 0) {
|
||||||
this.props.history.push(
|
this.branchSelected(defaultBranches[0]);
|
||||||
`${baseUrl}/${encodeURIComponent(defaultBranches[0].name)}/`
|
|
||||||
);
|
|
||||||
this.setState({ selectedBranch: defaultBranches[0] });
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
shouldRedirect = () => {
|
shouldRedirectToDefaultBranch = () => {
|
||||||
const { branches, revision } = this.props;
|
const { branches, revision } = this.props;
|
||||||
return branches && !revision;
|
return branches && !revision;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -2,9 +2,9 @@
|
|||||||
|
|
||||||
import { validation } from "@scm-manager/ui-components";
|
import { validation } from "@scm-manager/ui-components";
|
||||||
|
|
||||||
const { isNameValid, isMailValid } = validation;
|
const { isNameValid, isMailValid, isPathValid } = validation;
|
||||||
|
|
||||||
export { isNameValid, isMailValid };
|
export { isNameValid, isMailValid, isPathValid };
|
||||||
|
|
||||||
export const isDisplayNameValid = (displayName: string) => {
|
export const isDisplayNameValid = (displayName: string) => {
|
||||||
if (displayName) {
|
if (displayName) {
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import com.webcohesion.enunciate.metadata.rs.ResponseCode;
|
|||||||
import com.webcohesion.enunciate.metadata.rs.StatusCodes;
|
import com.webcohesion.enunciate.metadata.rs.StatusCodes;
|
||||||
import com.webcohesion.enunciate.metadata.rs.TypeHint;
|
import com.webcohesion.enunciate.metadata.rs.TypeHint;
|
||||||
import sonia.scm.plugin.AvailablePlugin;
|
import sonia.scm.plugin.AvailablePlugin;
|
||||||
import sonia.scm.plugin.InstalledPluginDescriptor;
|
import sonia.scm.plugin.InstalledPlugin;
|
||||||
import sonia.scm.plugin.PluginManager;
|
import sonia.scm.plugin.PluginManager;
|
||||||
import sonia.scm.plugin.PluginPermissions;
|
import sonia.scm.plugin.PluginPermissions;
|
||||||
import sonia.scm.web.VndMediaType;
|
import sonia.scm.web.VndMediaType;
|
||||||
@@ -19,6 +19,7 @@ import javax.ws.rs.QueryParam;
|
|||||||
import javax.ws.rs.core.Response;
|
import javax.ws.rs.core.Response;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import static sonia.scm.ContextEntry.ContextBuilder.entity;
|
import static sonia.scm.ContextEntry.ContextBuilder.entity;
|
||||||
import static sonia.scm.NotFoundException.notFound;
|
import static sonia.scm.NotFoundException.notFound;
|
||||||
@@ -51,10 +52,15 @@ public class AvailablePluginResource {
|
|||||||
@Produces(VndMediaType.PLUGIN_COLLECTION)
|
@Produces(VndMediaType.PLUGIN_COLLECTION)
|
||||||
public Response getAvailablePlugins() {
|
public Response getAvailablePlugins() {
|
||||||
PluginPermissions.read().check();
|
PluginPermissions.read().check();
|
||||||
List<AvailablePlugin> available = pluginManager.getAvailable();
|
List<InstalledPlugin> installed = pluginManager.getInstalled();
|
||||||
|
List<AvailablePlugin> available = pluginManager.getAvailable().stream().filter(a -> notInstalled(a, installed)).collect(Collectors.toList());
|
||||||
return Response.ok(collectionMapper.mapAvailable(available)).build();
|
return Response.ok(collectionMapper.mapAvailable(available)).build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean notInstalled(AvailablePlugin a, List<InstalledPlugin> installed) {
|
||||||
|
return installed.stream().noneMatch(installedPlugin -> installedPlugin.getDescriptor().getInformation().getName().equals(a.getDescriptor().getInformation().getName()));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns available plugin.
|
* Returns available plugin.
|
||||||
*
|
*
|
||||||
@@ -95,16 +101,4 @@ public class AvailablePluginResource {
|
|||||||
pluginManager.install(name, restartAfterInstallation);
|
pluginManager.install(name, restartAfterInstallation);
|
||||||
return Response.ok().build();
|
return Response.ok().build();
|
||||||
}
|
}
|
||||||
|
|
||||||
@POST
|
|
||||||
@Path("/install-pending")
|
|
||||||
@StatusCodes({
|
|
||||||
@ResponseCode(code = 200, condition = "success"),
|
|
||||||
@ResponseCode(code = 500, condition = "internal server error")
|
|
||||||
})
|
|
||||||
public Response installPending() {
|
|
||||||
PluginPermissions.manage().check();
|
|
||||||
pluginManager.installPendingAndRestart();
|
|
||||||
return Response.ok().build();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -54,6 +54,9 @@ public class IndexDtoGenerator extends HalAppenderMapper {
|
|||||||
builder.single(link("installedPlugins", resourceLinks.installedPluginCollection().self()));
|
builder.single(link("installedPlugins", resourceLinks.installedPluginCollection().self()));
|
||||||
builder.single(link("availablePlugins", resourceLinks.availablePluginCollection().self()));
|
builder.single(link("availablePlugins", resourceLinks.availablePluginCollection().self()));
|
||||||
}
|
}
|
||||||
|
if (PluginPermissions.manage().isPermitted()) {
|
||||||
|
builder.single(link("pendingPlugins", resourceLinks.pendingPluginCollection().self()));
|
||||||
|
}
|
||||||
if (UserPermissions.list().isPermitted()) {
|
if (UserPermissions.list().isPermitted()) {
|
||||||
builder.single(link("users", resourceLinks.userCollection().self()));
|
builder.single(link("users", resourceLinks.userCollection().self()));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,17 +3,19 @@ package sonia.scm.api.v2.resources;
|
|||||||
import com.webcohesion.enunciate.metadata.rs.ResponseCode;
|
import com.webcohesion.enunciate.metadata.rs.ResponseCode;
|
||||||
import com.webcohesion.enunciate.metadata.rs.StatusCodes;
|
import com.webcohesion.enunciate.metadata.rs.StatusCodes;
|
||||||
import com.webcohesion.enunciate.metadata.rs.TypeHint;
|
import com.webcohesion.enunciate.metadata.rs.TypeHint;
|
||||||
|
import sonia.scm.plugin.AvailablePlugin;
|
||||||
import sonia.scm.plugin.InstalledPlugin;
|
import sonia.scm.plugin.InstalledPlugin;
|
||||||
import sonia.scm.plugin.InstalledPluginDescriptor;
|
|
||||||
import sonia.scm.plugin.PluginManager;
|
import sonia.scm.plugin.PluginManager;
|
||||||
import sonia.scm.plugin.PluginPermissions;
|
import sonia.scm.plugin.PluginPermissions;
|
||||||
import sonia.scm.web.VndMediaType;
|
import sonia.scm.web.VndMediaType;
|
||||||
|
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
import javax.ws.rs.GET;
|
import javax.ws.rs.GET;
|
||||||
|
import javax.ws.rs.POST;
|
||||||
import javax.ws.rs.Path;
|
import javax.ws.rs.Path;
|
||||||
import javax.ws.rs.PathParam;
|
import javax.ws.rs.PathParam;
|
||||||
import javax.ws.rs.Produces;
|
import javax.ws.rs.Produces;
|
||||||
|
import javax.ws.rs.QueryParam;
|
||||||
import javax.ws.rs.core.Response;
|
import javax.ws.rs.core.Response;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
@@ -50,7 +52,8 @@ public class InstalledPluginResource {
|
|||||||
public Response getInstalledPlugins() {
|
public Response getInstalledPlugins() {
|
||||||
PluginPermissions.read().check();
|
PluginPermissions.read().check();
|
||||||
List<InstalledPlugin> plugins = pluginManager.getInstalled();
|
List<InstalledPlugin> plugins = pluginManager.getInstalled();
|
||||||
return Response.ok(collectionMapper.mapInstalled(plugins)).build();
|
List<AvailablePlugin> available = pluginManager.getAvailable();
|
||||||
|
return Response.ok(collectionMapper.mapInstalled(plugins, available)).build();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -72,10 +75,28 @@ public class InstalledPluginResource {
|
|||||||
public Response getInstalledPlugin(@PathParam("name") String name) {
|
public Response getInstalledPlugin(@PathParam("name") String name) {
|
||||||
PluginPermissions.read().check();
|
PluginPermissions.read().check();
|
||||||
Optional<InstalledPlugin> pluginDto = pluginManager.getInstalled(name);
|
Optional<InstalledPlugin> pluginDto = pluginManager.getInstalled(name);
|
||||||
|
List<AvailablePlugin> available = pluginManager.getAvailable();
|
||||||
if (pluginDto.isPresent()) {
|
if (pluginDto.isPresent()) {
|
||||||
return Response.ok(mapper.mapInstalled(pluginDto.get())).build();
|
return Response.ok(mapper.mapInstalled(pluginDto.get(), available)).build();
|
||||||
} else {
|
} else {
|
||||||
throw notFound(entity("Plugin", name));
|
throw notFound(entity("Plugin", name));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Triggers plugin uninstall.
|
||||||
|
* @param name plugin name
|
||||||
|
* @return HTTP Status.
|
||||||
|
*/
|
||||||
|
@POST
|
||||||
|
@Path("/{name}/uninstall")
|
||||||
|
@StatusCodes({
|
||||||
|
@ResponseCode(code = 200, condition = "success"),
|
||||||
|
@ResponseCode(code = 500, condition = "internal server error")
|
||||||
|
})
|
||||||
|
public Response uninstallPlugin(@PathParam("name") String name, @QueryParam("restart") boolean restartAfterInstallation) {
|
||||||
|
PluginPermissions.manage().check();
|
||||||
|
pluginManager.uninstall(name, restartAfterInstallation);
|
||||||
|
return Response.ok().build();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,113 @@
|
|||||||
|
package sonia.scm.api.v2.resources;
|
||||||
|
|
||||||
|
import com.webcohesion.enunciate.metadata.rs.ResponseCode;
|
||||||
|
import com.webcohesion.enunciate.metadata.rs.StatusCodes;
|
||||||
|
import de.otto.edison.hal.Embedded;
|
||||||
|
import de.otto.edison.hal.HalRepresentation;
|
||||||
|
import de.otto.edison.hal.Links;
|
||||||
|
import sonia.scm.plugin.AvailablePlugin;
|
||||||
|
import sonia.scm.plugin.InstalledPlugin;
|
||||||
|
import sonia.scm.plugin.PluginManager;
|
||||||
|
import sonia.scm.plugin.PluginPermissions;
|
||||||
|
import sonia.scm.web.VndMediaType;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
import javax.ws.rs.GET;
|
||||||
|
import javax.ws.rs.POST;
|
||||||
|
import javax.ws.rs.Path;
|
||||||
|
import javax.ws.rs.Produces;
|
||||||
|
import javax.ws.rs.core.Response;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
import static de.otto.edison.hal.Link.link;
|
||||||
|
import static de.otto.edison.hal.Links.linkingTo;
|
||||||
|
import static java.util.stream.Collectors.toList;
|
||||||
|
|
||||||
|
public class PendingPluginResource {
|
||||||
|
|
||||||
|
private final PluginManager pluginManager;
|
||||||
|
private final ResourceLinks resourceLinks;
|
||||||
|
private final PluginDtoMapper mapper;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public PendingPluginResource(PluginManager pluginManager, ResourceLinks resourceLinks, PluginDtoMapper mapper) {
|
||||||
|
this.pluginManager = pluginManager;
|
||||||
|
this.resourceLinks = resourceLinks;
|
||||||
|
this.mapper = mapper;
|
||||||
|
}
|
||||||
|
|
||||||
|
@GET
|
||||||
|
@Path("")
|
||||||
|
@StatusCodes({
|
||||||
|
@ResponseCode(code = 200, condition = "success"),
|
||||||
|
@ResponseCode(code = 500, condition = "internal server error")
|
||||||
|
})
|
||||||
|
@Produces(VndMediaType.PLUGIN_COLLECTION)
|
||||||
|
public Response getPending() {
|
||||||
|
PluginPermissions.manage().check();
|
||||||
|
|
||||||
|
List<AvailablePlugin> pending = pluginManager
|
||||||
|
.getAvailable()
|
||||||
|
.stream()
|
||||||
|
.filter(AvailablePlugin::isPending)
|
||||||
|
.collect(toList());
|
||||||
|
List<InstalledPlugin> installed = pluginManager.getInstalled();
|
||||||
|
|
||||||
|
Stream<AvailablePlugin> newPlugins = pending
|
||||||
|
.stream()
|
||||||
|
.filter(a -> !contains(installed, a));
|
||||||
|
Stream<InstalledPlugin> updatePlugins = installed
|
||||||
|
.stream()
|
||||||
|
.filter(i -> contains(pending, i));
|
||||||
|
Stream<InstalledPlugin> uninstallPlugins = installed
|
||||||
|
.stream()
|
||||||
|
.filter(InstalledPlugin::isMarkedForUninstall);
|
||||||
|
|
||||||
|
Links.Builder linksBuilder = linkingTo().self(resourceLinks.pendingPluginCollection().self());
|
||||||
|
|
||||||
|
List<PluginDto> installDtos = newPlugins.map(mapper::mapAvailable).collect(toList());
|
||||||
|
List<PluginDto> updateDtos = updatePlugins.map(i -> mapper.mapInstalled(i, pending)).collect(toList());
|
||||||
|
List<PluginDto> uninstallDtos = uninstallPlugins.map(i -> mapper.mapInstalled(i, pending)).collect(toList());
|
||||||
|
|
||||||
|
if (!installDtos.isEmpty() || !updateDtos.isEmpty() || !uninstallDtos.isEmpty()) {
|
||||||
|
linksBuilder.single(link("execute", resourceLinks.pendingPluginCollection().executePending()));
|
||||||
|
}
|
||||||
|
|
||||||
|
Embedded.Builder embedded = Embedded.embeddedBuilder();
|
||||||
|
embedded.with("new", installDtos);
|
||||||
|
embedded.with("update", updateDtos);
|
||||||
|
embedded.with("uninstall", uninstallDtos);
|
||||||
|
|
||||||
|
return Response.ok(new HalRepresentation(linksBuilder.build(), embedded.build())).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean contains(Collection<InstalledPlugin> installedPlugins, AvailablePlugin availablePlugin) {
|
||||||
|
return installedPlugins
|
||||||
|
.stream()
|
||||||
|
.anyMatch(installedPlugin -> haveSameName(installedPlugin, availablePlugin));
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean contains(Collection<AvailablePlugin> availablePlugins, InstalledPlugin installedPlugin) {
|
||||||
|
return availablePlugins
|
||||||
|
.stream()
|
||||||
|
.anyMatch(availablePlugin -> haveSameName(installedPlugin, availablePlugin));
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean haveSameName(InstalledPlugin installedPlugin, AvailablePlugin availablePlugin) {
|
||||||
|
return installedPlugin.getDescriptor().getInformation().getName().equals(availablePlugin.getDescriptor().getInformation().getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
@POST
|
||||||
|
@Path("/execute")
|
||||||
|
@StatusCodes({
|
||||||
|
@ResponseCode(code = 200, condition = "success"),
|
||||||
|
@ResponseCode(code = 500, condition = "internal server error")
|
||||||
|
})
|
||||||
|
public Response executePending() {
|
||||||
|
PluginPermissions.manage().check();
|
||||||
|
pluginManager.executePendingAndRestart();
|
||||||
|
return Response.ok().build();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
package sonia.scm.api.v2.resources;
|
package sonia.scm.api.v2.resources;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonInclude;
|
||||||
import de.otto.edison.hal.HalRepresentation;
|
import de.otto.edison.hal.HalRepresentation;
|
||||||
import de.otto.edison.hal.Links;
|
import de.otto.edison.hal.Links;
|
||||||
import lombok.Getter;
|
import lombok.Getter;
|
||||||
@@ -16,12 +17,17 @@ public class PluginDto extends HalRepresentation {
|
|||||||
|
|
||||||
private String name;
|
private String name;
|
||||||
private String version;
|
private String version;
|
||||||
|
@JsonInclude(JsonInclude.Include.NON_NULL)
|
||||||
|
private String newVersion;
|
||||||
private String displayName;
|
private String displayName;
|
||||||
private String description;
|
private String description;
|
||||||
private String author;
|
private String author;
|
||||||
private String category;
|
private String category;
|
||||||
private String avatarUrl;
|
private String avatarUrl;
|
||||||
private boolean pending;
|
private boolean pending;
|
||||||
|
@JsonInclude(JsonInclude.Include.NON_NULL)
|
||||||
|
private Boolean core;
|
||||||
|
private Boolean markedForUninstall;
|
||||||
private Set<String> dependencies;
|
private Set<String> dependencies;
|
||||||
|
|
||||||
public PluginDto(Links links) {
|
public PluginDto(Links links) {
|
||||||
|
|||||||
@@ -26,8 +26,11 @@ public class PluginDtoCollectionMapper {
|
|||||||
this.mapper = mapper;
|
this.mapper = mapper;
|
||||||
}
|
}
|
||||||
|
|
||||||
public HalRepresentation mapInstalled(List<InstalledPlugin> plugins) {
|
public HalRepresentation mapInstalled(List<InstalledPlugin> plugins, List<AvailablePlugin> availablePlugins) {
|
||||||
List<PluginDto> dtos = plugins.stream().map(mapper::mapInstalled).collect(toList());
|
List<PluginDto> dtos = plugins
|
||||||
|
.stream()
|
||||||
|
.map(i -> mapper.mapInstalled(i, availablePlugins))
|
||||||
|
.collect(toList());
|
||||||
return new HalRepresentation(createInstalledPluginsLinks(), embedDtos(dtos));
|
return new HalRepresentation(createInstalledPluginsLinks(), embedDtos(dtos));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,10 +53,6 @@ public class PluginDtoCollectionMapper {
|
|||||||
Links.Builder linksBuilder = linkingTo()
|
Links.Builder linksBuilder = linkingTo()
|
||||||
.with(Links.linkingTo().self(baseUrl).build());
|
.with(Links.linkingTo().self(baseUrl).build());
|
||||||
|
|
||||||
if (PluginPermissions.manage().isPermitted() && containsPending(plugins)) {
|
|
||||||
linksBuilder.single(Link.link("installPending", resourceLinks.availablePluginCollection().installPending()));
|
|
||||||
}
|
|
||||||
|
|
||||||
return linksBuilder.build();
|
return linksBuilder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -11,6 +11,9 @@ import sonia.scm.plugin.PluginPermissions;
|
|||||||
|
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
import static de.otto.edison.hal.Link.link;
|
import static de.otto.edison.hal.Link.link;
|
||||||
import static de.otto.edison.hal.Links.linkingTo;
|
import static de.otto.edison.hal.Links.linkingTo;
|
||||||
|
|
||||||
@@ -22,8 +25,8 @@ public abstract class PluginDtoMapper {
|
|||||||
|
|
||||||
public abstract void map(PluginInformation plugin, @MappingTarget PluginDto dto);
|
public abstract void map(PluginInformation plugin, @MappingTarget PluginDto dto);
|
||||||
|
|
||||||
public PluginDto mapInstalled(InstalledPlugin plugin) {
|
public PluginDto mapInstalled(InstalledPlugin plugin, List<AvailablePlugin> availablePlugins) {
|
||||||
PluginDto dto = createDtoForInstalled(plugin);
|
PluginDto dto = createDtoForInstalled(plugin, availablePlugins);
|
||||||
map(dto, plugin);
|
map(dto, plugin);
|
||||||
return dto;
|
return dto;
|
||||||
}
|
}
|
||||||
@@ -57,13 +60,43 @@ public abstract class PluginDtoMapper {
|
|||||||
return new PluginDto(links.build());
|
return new PluginDto(links.build());
|
||||||
}
|
}
|
||||||
|
|
||||||
private PluginDto createDtoForInstalled(InstalledPlugin plugin) {
|
private PluginDto createDtoForInstalled(InstalledPlugin plugin, List<AvailablePlugin> availablePlugins) {
|
||||||
PluginInformation information = plugin.getDescriptor().getInformation();
|
PluginInformation information = plugin.getDescriptor().getInformation();
|
||||||
|
Optional<AvailablePlugin> availablePlugin = checkForUpdates(plugin, availablePlugins);
|
||||||
|
|
||||||
Links.Builder links = linkingTo()
|
Links.Builder links = linkingTo()
|
||||||
.self(resourceLinks.installedPlugin()
|
.self(resourceLinks.installedPlugin()
|
||||||
.self(information.getName()));
|
.self(information.getName()));
|
||||||
|
if (!plugin.isCore()
|
||||||
|
&& availablePlugin.isPresent()
|
||||||
|
&& !availablePlugin.get().isPending()
|
||||||
|
&& PluginPermissions.manage().isPermitted()
|
||||||
|
) {
|
||||||
|
links.single(link("update", resourceLinks.availablePlugin().install(information.getName())));
|
||||||
|
}
|
||||||
|
if (plugin.isUninstallable()
|
||||||
|
&& (!availablePlugin.isPresent() || !availablePlugin.get().isPending())
|
||||||
|
&& PluginPermissions.manage().isPermitted()
|
||||||
|
) {
|
||||||
|
links.single(link("uninstall", resourceLinks.installedPlugin().uninstall(information.getName())));
|
||||||
|
}
|
||||||
|
|
||||||
return new PluginDto(links.build());
|
PluginDto dto = new PluginDto(links.build());
|
||||||
|
|
||||||
|
availablePlugin.ifPresent(value -> {
|
||||||
|
dto.setNewVersion(value.getDescriptor().getInformation().getVersion());
|
||||||
|
dto.setPending(value.isPending());
|
||||||
|
});
|
||||||
|
|
||||||
|
dto.setCore(plugin.isCore());
|
||||||
|
dto.setMarkedForUninstall(plugin.isMarkedForUninstall());
|
||||||
|
|
||||||
|
return dto;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Optional<AvailablePlugin> checkForUpdates(InstalledPlugin plugin, List<AvailablePlugin> availablePlugins) {
|
||||||
|
return availablePlugins.stream()
|
||||||
|
.filter(a -> a.getDescriptor().getInformation().getName().equals(plugin.getDescriptor().getInformation().getName()))
|
||||||
|
.findAny();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,11 +9,13 @@ public class PluginRootResource {
|
|||||||
|
|
||||||
private Provider<InstalledPluginResource> installedPluginResourceProvider;
|
private Provider<InstalledPluginResource> installedPluginResourceProvider;
|
||||||
private Provider<AvailablePluginResource> availablePluginResourceProvider;
|
private Provider<AvailablePluginResource> availablePluginResourceProvider;
|
||||||
|
private Provider<PendingPluginResource> pendingPluginResourceProvider;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public PluginRootResource(Provider<InstalledPluginResource> installedPluginResourceProvider, Provider<AvailablePluginResource> availablePluginResourceProvider) {
|
public PluginRootResource(Provider<InstalledPluginResource> installedPluginResourceProvider, Provider<AvailablePluginResource> availablePluginResourceProvider, Provider<PendingPluginResource> pendingPluginResourceProvider) {
|
||||||
this.installedPluginResourceProvider = installedPluginResourceProvider;
|
this.installedPluginResourceProvider = installedPluginResourceProvider;
|
||||||
this.availablePluginResourceProvider = availablePluginResourceProvider;
|
this.availablePluginResourceProvider = availablePluginResourceProvider;
|
||||||
|
this.pendingPluginResourceProvider = pendingPluginResourceProvider;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Path("/installed")
|
@Path("/installed")
|
||||||
@@ -23,4 +25,7 @@ public class PluginRootResource {
|
|||||||
|
|
||||||
@Path("/available")
|
@Path("/available")
|
||||||
public AvailablePluginResource availablePlugins() { return availablePluginResourceProvider.get(); }
|
public AvailablePluginResource availablePlugins() { return availablePluginResourceProvider.get(); }
|
||||||
|
|
||||||
|
@Path("/pending")
|
||||||
|
public PendingPluginResource pendingPlugins() { return pendingPluginResourceProvider.get(); }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -666,6 +666,10 @@ class ResourceLinks {
|
|||||||
String self(String id) {
|
String self(String id) {
|
||||||
return installedPluginLinkBuilder.method("installedPlugins").parameters().method("getInstalledPlugin").parameters(id).href();
|
return installedPluginLinkBuilder.method("installedPlugins").parameters().method("getInstalledPlugin").parameters(id).href();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String uninstall(String name) {
|
||||||
|
return installedPluginLinkBuilder.method("installedPlugins").parameters().method("uninstallPlugin").parameters(name).href();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public InstalledPluginCollectionLinks installedPluginCollection() {
|
public InstalledPluginCollectionLinks installedPluginCollection() {
|
||||||
@@ -715,12 +719,28 @@ class ResourceLinks {
|
|||||||
availablePluginCollectionLinkBuilder = new LinkBuilder(pathInfo, PluginRootResource.class, AvailablePluginResource.class);
|
availablePluginCollectionLinkBuilder = new LinkBuilder(pathInfo, PluginRootResource.class, AvailablePluginResource.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
String installPending() {
|
String self() {
|
||||||
return availablePluginCollectionLinkBuilder.method("availablePlugins").parameters().method("installPending").parameters().href();
|
return availablePluginCollectionLinkBuilder.method("availablePlugins").parameters().method("getAvailablePlugins").parameters().href();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public PendingPluginCollectionLinks pendingPluginCollection() {
|
||||||
|
return new PendingPluginCollectionLinks(scmPathInfoStore.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
static class PendingPluginCollectionLinks {
|
||||||
|
private final LinkBuilder pendingPluginCollectionLinkBuilder;
|
||||||
|
|
||||||
|
PendingPluginCollectionLinks(ScmPathInfo pathInfo) {
|
||||||
|
pendingPluginCollectionLinkBuilder = new LinkBuilder(pathInfo, PluginRootResource.class, PendingPluginResource.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
String executePending() {
|
||||||
|
return pendingPluginCollectionLinkBuilder.method("pendingPlugins").parameters().method("executePending").parameters().href();
|
||||||
}
|
}
|
||||||
|
|
||||||
String self() {
|
String self() {
|
||||||
return availablePluginCollectionLinkBuilder.method("availablePlugins").parameters().method("getAvailablePlugins").parameters().href();
|
return pendingPluginCollectionLinkBuilder.method("pendingPlugins").parameters().method("getPending").parameters().href();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -29,9 +29,11 @@ import java.io.File;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.MalformedURLException;
|
import java.net.MalformedURLException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
|
import java.nio.file.Path;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
public final class PluginBootstrap {
|
public final class PluginBootstrap {
|
||||||
|
|
||||||
@@ -78,12 +80,37 @@ public final class PluginBootstrap {
|
|||||||
LOG.info("core plugin extraction is disabled");
|
LOG.info("core plugin extraction is disabled");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uninstallMarkedPlugins(pluginDirectory.toPath());
|
||||||
return PluginsInternal.collectPlugins(classLoaderLifeCycle, pluginDirectory.toPath());
|
return PluginsInternal.collectPlugins(classLoaderLifeCycle, pluginDirectory.toPath());
|
||||||
} catch (IOException ex) {
|
} catch (IOException ex) {
|
||||||
throw new PluginLoadException("could not load plugins", ex);
|
throw new PluginLoadException("could not load plugins", ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void uninstallMarkedPlugins(Path pluginDirectory) {
|
||||||
|
try (Stream<Path> list = java.nio.file.Files.list(pluginDirectory)) {
|
||||||
|
list
|
||||||
|
.filter(java.nio.file.Files::isDirectory)
|
||||||
|
.filter(this::isMarkedForUninstall)
|
||||||
|
.forEach(this::uninstall);
|
||||||
|
} catch (IOException e) {
|
||||||
|
LOG.warn("error occurred while checking for plugins that should be uninstalled", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isMarkedForUninstall(Path path) {
|
||||||
|
return java.nio.file.Files.exists(path.resolve(InstalledPlugin.UNINSTALL_MARKER_FILENAME));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void uninstall(Path path) {
|
||||||
|
try {
|
||||||
|
LOG.info("deleting plugin directory {}", path);
|
||||||
|
IOUtil.delete(path.toFile());
|
||||||
|
} catch (IOException e) {
|
||||||
|
LOG.warn("could not delete plugin directory {}", path, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void renameOldPluginsFolder(File pluginDirectory) {
|
private void renameOldPluginsFolder(File pluginDirectory) {
|
||||||
if (new File(pluginDirectory, "classpath.xml").exists()) {
|
if (new File(pluginDirectory, "classpath.xml").exists()) {
|
||||||
File backupDirectory = new File(pluginDirectory.getParentFile(), "plugins.v1");
|
File backupDirectory = new File(pluginDirectory.getParentFile(), "plugins.v1");
|
||||||
@@ -96,7 +123,6 @@ public final class PluginBootstrap {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private boolean isCorePluginExtractionDisabled() {
|
private boolean isCorePluginExtractionDisabled() {
|
||||||
return Boolean.getBoolean("sonia.scm.boot.disable-core-plugin-extraction");
|
return Boolean.getBoolean("sonia.scm.boot.disable-core-plugin-extraction");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,8 +33,7 @@
|
|||||||
|
|
||||||
package sonia.scm.plugin;
|
package sonia.scm.plugin;
|
||||||
|
|
||||||
//~--- non-JDK imports --------------------------------------------------------
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
import com.google.inject.Singleton;
|
import com.google.inject.Singleton;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
@@ -42,10 +41,13 @@ import org.slf4j.LoggerFactory;
|
|||||||
import sonia.scm.NotFoundException;
|
import sonia.scm.NotFoundException;
|
||||||
import sonia.scm.event.ScmEventBus;
|
import sonia.scm.event.ScmEventBus;
|
||||||
import sonia.scm.lifecycle.RestartEvent;
|
import sonia.scm.lifecycle.RestartEvent;
|
||||||
|
import sonia.scm.version.Version;
|
||||||
|
|
||||||
//~--- JDK imports ------------------------------------------------------------
|
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.file.Files;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
@@ -53,6 +55,9 @@ import java.util.function.Predicate;
|
|||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import static sonia.scm.ContextEntry.ContextBuilder.entity;
|
import static sonia.scm.ContextEntry.ContextBuilder.entity;
|
||||||
|
import static sonia.scm.ScmConstraintViolationException.Builder.doThrow;
|
||||||
|
|
||||||
|
//~--- JDK imports ------------------------------------------------------------
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
@@ -67,7 +72,8 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
private final PluginLoader loader;
|
private final PluginLoader loader;
|
||||||
private final PluginCenter center;
|
private final PluginCenter center;
|
||||||
private final PluginInstaller installer;
|
private final PluginInstaller installer;
|
||||||
private final List<PendingPluginInstallation> pendingQueue = new ArrayList<>();
|
private final Collection<PendingPluginInstallation> pendingQueue = new ArrayList<>();
|
||||||
|
private final PluginDependencyTracker dependencyTracker = new PluginDependencyTracker();
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public DefaultPluginManager(ScmEventBus eventBus, PluginLoader loader, PluginCenter center, PluginInstaller installer) {
|
public DefaultPluginManager(ScmEventBus eventBus, PluginLoader loader, PluginCenter center, PluginInstaller installer) {
|
||||||
@@ -75,6 +81,17 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
this.loader = loader;
|
this.loader = loader;
|
||||||
this.center = center;
|
this.center = center;
|
||||||
this.installer = installer;
|
this.installer = installer;
|
||||||
|
|
||||||
|
this.computeInstallationDependencies();
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
synchronized void computeInstallationDependencies() {
|
||||||
|
loader.getInstalledPlugins()
|
||||||
|
.stream()
|
||||||
|
.map(InstalledPlugin::getDescriptor)
|
||||||
|
.forEach(dependencyTracker::addInstalled);
|
||||||
|
updateMayUninstallFlag();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -83,7 +100,7 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
return center.getAvailable()
|
return center.getAvailable()
|
||||||
.stream()
|
.stream()
|
||||||
.filter(filterByName(name))
|
.filter(filterByName(name))
|
||||||
.filter(this::isNotInstalled)
|
.filter(this::isNotInstalledOrMoreUpToDate)
|
||||||
.map(p -> getPending(name).orElse(p))
|
.map(p -> getPending(name).orElse(p))
|
||||||
.findFirst();
|
.findFirst();
|
||||||
}
|
}
|
||||||
@@ -116,7 +133,7 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
PluginPermissions.read().check();
|
PluginPermissions.read().check();
|
||||||
return center.getAvailable()
|
return center.getAvailable()
|
||||||
.stream()
|
.stream()
|
||||||
.filter(this::isNotInstalled)
|
.filter(this::isNotInstalledOrMoreUpToDate)
|
||||||
.map(p -> getPending(p.getDescriptor().getInformation().getName()).orElse(p))
|
.map(p -> getPending(p.getDescriptor().getInformation().getName()).orElse(p))
|
||||||
.collect(Collectors.toList());
|
.collect(Collectors.toList());
|
||||||
}
|
}
|
||||||
@@ -125,18 +142,32 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
return plugin -> name.equals(plugin.getDescriptor().getInformation().getName());
|
return plugin -> name.equals(plugin.getDescriptor().getInformation().getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isNotInstalled(AvailablePlugin availablePlugin) {
|
private boolean isNotInstalledOrMoreUpToDate(AvailablePlugin availablePlugin) {
|
||||||
return !getInstalled(availablePlugin.getDescriptor().getInformation().getName()).isPresent();
|
return getInstalled(availablePlugin.getDescriptor().getInformation().getName())
|
||||||
|
.map(installedPlugin -> availableIsMoreUpToDateThanInstalled(availablePlugin, installedPlugin))
|
||||||
|
.orElse(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean availableIsMoreUpToDateThanInstalled(AvailablePlugin availablePlugin, InstalledPlugin installed) {
|
||||||
|
return Version.parse(availablePlugin.getDescriptor().getInformation().getVersion()).isNewer(installed.getDescriptor().getInformation().getVersion());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void install(String name, boolean restartAfterInstallation) {
|
public void install(String name, boolean restartAfterInstallation) {
|
||||||
PluginPermissions.manage().check();
|
PluginPermissions.manage().check();
|
||||||
|
|
||||||
|
getInstalled(name)
|
||||||
|
.map(InstalledPlugin::isCore)
|
||||||
|
.ifPresent(
|
||||||
|
core -> doThrow().violation("plugin is a core plugin and cannot be updated").when(core)
|
||||||
|
);
|
||||||
|
|
||||||
List<AvailablePlugin> plugins = collectPluginsToInstall(name);
|
List<AvailablePlugin> plugins = collectPluginsToInstall(name);
|
||||||
List<PendingPluginInstallation> pendingInstallations = new ArrayList<>();
|
List<PendingPluginInstallation> pendingInstallations = new ArrayList<>();
|
||||||
for (AvailablePlugin plugin : plugins) {
|
for (AvailablePlugin plugin : plugins) {
|
||||||
try {
|
try {
|
||||||
PendingPluginInstallation pending = installer.install(plugin);
|
PendingPluginInstallation pending = installer.install(plugin);
|
||||||
|
dependencyTracker.addInstalled(plugin.getDescriptor());
|
||||||
pendingInstallations.add(pending);
|
pendingInstallations.add(pending);
|
||||||
} catch (PluginInstallException ex) {
|
} catch (PluginInstallException ex) {
|
||||||
cancelPending(pendingInstallations);
|
cancelPending(pendingInstallations);
|
||||||
@@ -149,15 +180,54 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
restart("plugin installation");
|
restart("plugin installation");
|
||||||
} else {
|
} else {
|
||||||
pendingQueue.addAll(pendingInstallations);
|
pendingQueue.addAll(pendingInstallations);
|
||||||
|
updateMayUninstallFlag();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void installPendingAndRestart() {
|
public void uninstall(String name, boolean restartAfterInstallation) {
|
||||||
PluginPermissions.manage().check();
|
PluginPermissions.manage().check();
|
||||||
if (!pendingQueue.isEmpty()) {
|
InstalledPlugin installed = getInstalled(name)
|
||||||
restart("install pending plugins");
|
.orElseThrow(() -> NotFoundException.notFound(entity(InstalledPlugin.class, name)));
|
||||||
|
doThrow().violation("plugin is a core plugin and cannot be uninstalled").when(installed.isCore());
|
||||||
|
|
||||||
|
dependencyTracker.removeInstalled(installed.getDescriptor());
|
||||||
|
installed.setMarkedForUninstall(true);
|
||||||
|
|
||||||
|
createMarkerFile(installed, InstalledPlugin.UNINSTALL_MARKER_FILENAME);
|
||||||
|
|
||||||
|
if (restartAfterInstallation) {
|
||||||
|
restart("plugin installation");
|
||||||
|
} else {
|
||||||
|
updateMayUninstallFlag();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateMayUninstallFlag() {
|
||||||
|
loader.getInstalledPlugins()
|
||||||
|
.forEach(p -> p.setUninstallable(isUninstallable(p)));
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isUninstallable(InstalledPlugin p) {
|
||||||
|
return !p.isCore()
|
||||||
|
&& !p.isMarkedForUninstall()
|
||||||
|
&& dependencyTracker.mayUninstall(p.getDescriptor().getInformation().getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
private void createMarkerFile(InstalledPlugin plugin, String markerFile) {
|
||||||
|
try {
|
||||||
|
Files.createFile(plugin.getDirectory().resolve(markerFile));
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new PluginException("could not mark plugin " + plugin.getId() + " in path " + plugin.getDirectory() + "as " + markerFile, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void executePendingAndRestart() {
|
||||||
|
PluginPermissions.manage().check();
|
||||||
|
if (!pendingQueue.isEmpty() || getInstalled().stream().anyMatch(InstalledPlugin::isMarkedForUninstall)) {
|
||||||
|
restart("execute pending plugin changes");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -171,22 +241,18 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
|
|
||||||
private List<AvailablePlugin> collectPluginsToInstall(String name) {
|
private List<AvailablePlugin> collectPluginsToInstall(String name) {
|
||||||
List<AvailablePlugin> plugins = new ArrayList<>();
|
List<AvailablePlugin> plugins = new ArrayList<>();
|
||||||
collectPluginsToInstall(plugins, name);
|
collectPluginsToInstall(plugins, name, true);
|
||||||
return plugins;
|
return plugins;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isInstalledOrPending(String name) {
|
private void collectPluginsToInstall(List<AvailablePlugin> plugins, String name, boolean isUpdate) {
|
||||||
return getInstalled(name).isPresent() || getPending(name).isPresent();
|
if (!isInstalledOrPending(name) || isUpdate && isUpdatable(name)) {
|
||||||
}
|
|
||||||
|
|
||||||
private void collectPluginsToInstall(List<AvailablePlugin> plugins, String name) {
|
|
||||||
if (!isInstalledOrPending(name)) {
|
|
||||||
AvailablePlugin plugin = getAvailable(name).orElseThrow(() -> NotFoundException.notFound(entity(AvailablePlugin.class, name)));
|
AvailablePlugin plugin = getAvailable(name).orElseThrow(() -> NotFoundException.notFound(entity(AvailablePlugin.class, name)));
|
||||||
|
|
||||||
Set<String> dependencies = plugin.getDescriptor().getDependencies();
|
Set<String> dependencies = plugin.getDescriptor().getDependencies();
|
||||||
if (dependencies != null) {
|
if (dependencies != null) {
|
||||||
for (String dependency: dependencies){
|
for (String dependency: dependencies){
|
||||||
collectPluginsToInstall(plugins, dependency);
|
collectPluginsToInstall(plugins, dependency, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -195,4 +261,12 @@ public class DefaultPluginManager implements PluginManager {
|
|||||||
LOG.info("plugin {} is already installed or installation is pending, skipping installation", name);
|
LOG.info("plugin {} is already installed or installation is pending, skipping installation", name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean isInstalledOrPending(String name) {
|
||||||
|
return getInstalled(name).isPresent() || getPending(name).isPresent();
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isUpdatable(String name) {
|
||||||
|
return getAvailable(name).isPresent() && !getPending(name).isPresent();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,8 +33,8 @@ class PluginCenterLoader {
|
|||||||
LOG.info("fetch plugins from {}", url);
|
LOG.info("fetch plugins from {}", url);
|
||||||
PluginCenterDto pluginCenterDto = client.get(url).request().contentFromJson(PluginCenterDto.class);
|
PluginCenterDto pluginCenterDto = client.get(url).request().contentFromJson(PluginCenterDto.class);
|
||||||
return mapper.map(pluginCenterDto);
|
return mapper.map(pluginCenterDto);
|
||||||
} catch (IOException ex) {
|
} catch (Exception ex) {
|
||||||
LOG.error("failed to load plugins from plugin center, returning empty list");
|
LOG.error("failed to load plugins from plugin center, returning empty list", ex);
|
||||||
return Collections.emptySet();
|
return Collections.emptySet();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package sonia.scm.plugin;
|
||||||
|
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import static sonia.scm.ScmConstraintViolationException.Builder.doThrow;
|
||||||
|
|
||||||
|
class PluginDependencyTracker {
|
||||||
|
|
||||||
|
private final Map<String, Collection<String>> plugins = new HashMap<>();
|
||||||
|
|
||||||
|
void addInstalled(PluginDescriptor plugin) {
|
||||||
|
if (plugin.getDependencies() != null) {
|
||||||
|
plugin.getDependencies().forEach(dependency -> addDependency(plugin.getInformation().getName(), dependency));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void removeInstalled(PluginDescriptor plugin) {
|
||||||
|
doThrow()
|
||||||
|
.violation("Plugin is needed as a dependency for other plugins", "plugin")
|
||||||
|
.when(!mayUninstall(plugin.getInformation().getName()));
|
||||||
|
plugin.getDependencies().forEach(dependency -> removeDependency(plugin.getInformation().getName(), dependency));
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean mayUninstall(String name) {
|
||||||
|
return plugins.computeIfAbsent(name, x -> new HashSet<>()).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addDependency(String from, String to) {
|
||||||
|
plugins.computeIfAbsent(to, name -> new HashSet<>()).add(from);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void removeDependency(String from, String to) {
|
||||||
|
plugins.get(to).remove(from);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -461,13 +461,16 @@ public final class PluginProcessor
|
|||||||
Path descriptorPath = directory.resolve(PluginConstants.FILE_DESCRIPTOR);
|
Path descriptorPath = directory.resolve(PluginConstants.FILE_DESCRIPTOR);
|
||||||
|
|
||||||
if (Files.exists(descriptorPath)) {
|
if (Files.exists(descriptorPath)) {
|
||||||
|
|
||||||
|
boolean core = Files.exists(directory.resolve(PluginConstants.FILE_CORE));
|
||||||
|
|
||||||
ClassLoader cl = createClassLoader(classLoader, smp);
|
ClassLoader cl = createClassLoader(classLoader, smp);
|
||||||
|
|
||||||
InstalledPluginDescriptor descriptor = createDescriptor(cl, descriptorPath);
|
InstalledPluginDescriptor descriptor = createDescriptor(cl, descriptorPath);
|
||||||
|
|
||||||
WebResourceLoader resourceLoader = createWebResourceLoader(directory);
|
WebResourceLoader resourceLoader = createWebResourceLoader(directory);
|
||||||
|
|
||||||
plugin = new InstalledPlugin(descriptor, cl, resourceLoader, directory);
|
plugin = new InstalledPlugin(descriptor, cl, resourceLoader, directory, core);
|
||||||
} else {
|
} else {
|
||||||
logger.warn("found plugin directory without plugin descriptor");
|
logger.warn("found plugin directory without plugin descriptor");
|
||||||
}
|
}
|
||||||
|
|||||||
186
scm-webapp/src/main/resources/locales/es/plugins.json
Normal file
186
scm-webapp/src/main/resources/locales/es/plugins.json
Normal file
@@ -0,0 +1,186 @@
|
|||||||
|
{
|
||||||
|
"permissions": {
|
||||||
|
"*": {
|
||||||
|
"displayName": "Administrador global",
|
||||||
|
"description": "Puede administrar la instancia completa"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"read,pull": {
|
||||||
|
"*": {
|
||||||
|
"displayName": "Leer todos los repositorios",
|
||||||
|
"description": "Puede ver y clonar todos los repositorios"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"read,pull,push": {
|
||||||
|
"*": {
|
||||||
|
"displayName": "Escribir todos los repositorios",
|
||||||
|
"description": "Puede ver, clonar y enviar cambios a todos los repositorios"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"*": {
|
||||||
|
"displayName": "Poseer todos los repositorios",
|
||||||
|
"description": "Puede ver, clonar, enviar cambios, configurar y eliminar todos los repositorios"
|
||||||
|
},
|
||||||
|
"create": {
|
||||||
|
"displayName": "Crear repositorios",
|
||||||
|
"description": "Puede crear repositorios"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"user": {
|
||||||
|
"*": {
|
||||||
|
"displayName": "Administrar usuarios",
|
||||||
|
"description": "Puede administrar todos los usuarios"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"group": {
|
||||||
|
"*": {
|
||||||
|
"displayName": "Administrar grupos",
|
||||||
|
"description": "Puede administrar todos los grupos"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"permission": {
|
||||||
|
"*": {
|
||||||
|
"displayName": "Administrar permisos",
|
||||||
|
"description": "Puede administrar permisos (addicionalmente necesita 'administrar usuarios' y/o 'administrar grupos')."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"configuration": {
|
||||||
|
"list": {
|
||||||
|
"displayName": "Administración basíca",
|
||||||
|
"description": "Prerequisito para todos los permisos administrativos. Sin este permiso la configuración no será visible."
|
||||||
|
},
|
||||||
|
"read,write": {
|
||||||
|
"global": {
|
||||||
|
"displayName": "Administrar núcleo",
|
||||||
|
"description": "Puede administrar las opciones centrales de SCM-Manager"
|
||||||
|
},
|
||||||
|
"*": {
|
||||||
|
"displayName": "Administrar núcleo y complementos",
|
||||||
|
"description": "Puede configurar las opciones centrales de SCM-Manager y todos los complementos (plugins)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"repositoryRole": {
|
||||||
|
"read,write": {
|
||||||
|
"displayName": "Administrar permisos de roles de repositorio personalizados",
|
||||||
|
"description": "Puede crear, modificar y borrar roles de repositorios personalizados y sus permisos"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"plugin": {
|
||||||
|
"read": {
|
||||||
|
"displayName": "Leer todos los complementos",
|
||||||
|
"description": "Puede leer todos los complementos instalados y los disponibles"
|
||||||
|
},
|
||||||
|
"read,write": {
|
||||||
|
"displayName": "Leer y gestionar todos los complementos",
|
||||||
|
"description": "Puede leer y gestionar todos los complementos instalados y los diponibles"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"unknown": "Permiso desconocido"
|
||||||
|
},
|
||||||
|
"verbs": {
|
||||||
|
"repository": {
|
||||||
|
"read": {
|
||||||
|
"displayName": "leer repositorio",
|
||||||
|
"description": "Puede leer el repositorio dentro de SCM-Manager"
|
||||||
|
},
|
||||||
|
"modify": {
|
||||||
|
"displayName": "modificar los metadatos del repositorio",
|
||||||
|
"description": "Puede modificar las propiedades básicas del repositorio"
|
||||||
|
},
|
||||||
|
"delete": {
|
||||||
|
"displayName": "borrar repositorio",
|
||||||
|
"description": "Puede borrar el repositorio"
|
||||||
|
},
|
||||||
|
"pull": {
|
||||||
|
"displayName": "Traer los cambios o crear copia local del repositorio (pull/checkout)",
|
||||||
|
"description": "Puede traer los cambios o crear una copia de trabajo local desde el repositorio (pull/checkout)"
|
||||||
|
},
|
||||||
|
"push": {
|
||||||
|
"displayName": "Publicar o enviar cambios al repositorio (push/commit)",
|
||||||
|
"description": "Puede cambiar el contenido del repositorio (push/commit)"
|
||||||
|
},
|
||||||
|
"permissionRead": {
|
||||||
|
"displayName": "leer permisos",
|
||||||
|
"description": "Puede ver los permisos del repositorio"
|
||||||
|
},
|
||||||
|
"permissionWrite": {
|
||||||
|
"displayName": "modificar permisos",
|
||||||
|
"description": "Puede modificar los permisos del repositorio"
|
||||||
|
},
|
||||||
|
"*": {
|
||||||
|
"displayName": "poseer repositorio",
|
||||||
|
"description": "Puede cambiar todo en el repositorio (incluidos los demás permisos)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"errors": {
|
||||||
|
"context": "Contexto",
|
||||||
|
"errorCode": "Código de error",
|
||||||
|
"transactionId": "Identificador de la transacción",
|
||||||
|
"moreInfo": "Para más información ver",
|
||||||
|
"violations": "Violaciones:",
|
||||||
|
"AGR7UzkhA1": {
|
||||||
|
"displayName": "No encontrado",
|
||||||
|
"description": "No se pudo encontrar la entidad solicitada. Puede haber sido eliminado en otra sesión"
|
||||||
|
},
|
||||||
|
"FtR7UznKU1": {
|
||||||
|
"displayName": "Ya existe",
|
||||||
|
"description": "Ya hay una entidad con el mismo valor de clave."
|
||||||
|
},
|
||||||
|
"9BR7qpDAe1": {
|
||||||
|
"displayName": "No se permite cambiar la contraseña",
|
||||||
|
"description": "No tiene permiso para cambiar la contraseña."
|
||||||
|
},
|
||||||
|
"2wR7UzpPG1": {
|
||||||
|
"displayName": "Modificaciones actuales",
|
||||||
|
"description": "La entidad ha sido modificada concurrentemente por otro usuario o proceso. Por favor recarge la entidad."
|
||||||
|
},
|
||||||
|
"9SR8G0kmU1": {
|
||||||
|
"displayName": "Funcionalidad no soportada",
|
||||||
|
"description": "El sistema de control de versiones de este repositorio no admite la funcionalidad solicitada."
|
||||||
|
},
|
||||||
|
"CmR8GCJb31": {
|
||||||
|
"displayName": "Error interno del servidor",
|
||||||
|
"description": "Se ha producido un error interno en el servidor. Por favor contacte con su administrador para obtener más ayuda."
|
||||||
|
},
|
||||||
|
"92RCCCMHO1": {
|
||||||
|
"displayName": "No se ha podido encontrar la URL interna",
|
||||||
|
"description": "Una petición interna no ha podido ser manejada por el servidor. Por favor contacte con su administrador para obtener más ayuda."
|
||||||
|
},
|
||||||
|
"2VRCrvpL71": {
|
||||||
|
"displayName": "Formato de datos invorrecto",
|
||||||
|
"description": "Los datos enviados al servidor son son correctos. Por favor revise los datos enviados o contacte con su administrador para obtener más ayuda."
|
||||||
|
},
|
||||||
|
"8pRBYDURx1": {
|
||||||
|
"displayName": "Tipo de datos incorrecto",
|
||||||
|
"description": "El tipo de los datos enviados el servidor es incorrecto. Por favor contacte con su administrador para obtener más ayuda."
|
||||||
|
},
|
||||||
|
"1wR7ZBe7H1": {
|
||||||
|
"displayName": "Entrada incorrecta",
|
||||||
|
"description": "Los datos no pueden ser validados. Por favor corríjalos e inténtelo de nuevo."
|
||||||
|
},
|
||||||
|
"3zR9vPNIE1": {
|
||||||
|
"displayName": "Entrada incorrecta",
|
||||||
|
"description": "Los datos no pueden ser validados. Por favor corríjalos e inténtelo de nuevo."
|
||||||
|
},
|
||||||
|
"CHRM7IQzo1": {
|
||||||
|
"displayName": "Cambio fallido",
|
||||||
|
"description": "El cambio ha fallado. Por favor contacte con su administrador para obtener más ayuda."
|
||||||
|
},
|
||||||
|
"thbsUFokjk": {
|
||||||
|
"displayName": "Cambio inválido en el identificador",
|
||||||
|
"description": "Un identificador ha sido cambiado en la entidad. Esto no está permitido."
|
||||||
|
},
|
||||||
|
"40RaYIeeR1": {
|
||||||
|
"displayName": "No se han efectuado modificaciones.",
|
||||||
|
"description": "Ningún fichero del repositorio ha sido modificado. Por lo tanto, no se confirmará ninguna modificación."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"namespaceStrategies": {
|
||||||
|
"UsernameNamespaceStrategy": "Nombre de usuario",
|
||||||
|
"CustomNamespaceStrategy": "Personalizar",
|
||||||
|
"CurrentYearNamespaceStrategy": "Año actual",
|
||||||
|
"RepositoryTypeNamespaceStrategy": "Tipo de repositorio"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
package sonia.scm.api.v2.resources;
|
package sonia.scm.api.v2.resources;
|
||||||
|
|
||||||
import de.otto.edison.hal.HalRepresentation;
|
import de.otto.edison.hal.HalRepresentation;
|
||||||
|
import org.apache.shiro.ShiroException;
|
||||||
import org.apache.shiro.subject.Subject;
|
import org.apache.shiro.subject.Subject;
|
||||||
import org.apache.shiro.util.ThreadContext;
|
import org.apache.shiro.util.ThreadContext;
|
||||||
import org.jboss.resteasy.core.Dispatcher;
|
import org.jboss.resteasy.core.Dispatcher;
|
||||||
@@ -18,6 +19,8 @@ import org.mockito.Mock;
|
|||||||
import org.mockito.junit.jupiter.MockitoExtension;
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
import sonia.scm.plugin.AvailablePlugin;
|
import sonia.scm.plugin.AvailablePlugin;
|
||||||
import sonia.scm.plugin.AvailablePluginDescriptor;
|
import sonia.scm.plugin.AvailablePluginDescriptor;
|
||||||
|
import sonia.scm.plugin.InstalledPlugin;
|
||||||
|
import sonia.scm.plugin.InstalledPluginDescriptor;
|
||||||
import sonia.scm.plugin.PluginCondition;
|
import sonia.scm.plugin.PluginCondition;
|
||||||
import sonia.scm.plugin.PluginInformation;
|
import sonia.scm.plugin.PluginInformation;
|
||||||
import sonia.scm.plugin.PluginManager;
|
import sonia.scm.plugin.PluginManager;
|
||||||
@@ -25,7 +28,6 @@ import sonia.scm.web.VndMediaType;
|
|||||||
|
|
||||||
import javax.inject.Provider;
|
import javax.inject.Provider;
|
||||||
import javax.servlet.http.HttpServletResponse;
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
|
||||||
import java.io.UnsupportedEncodingException;
|
import java.io.UnsupportedEncodingException;
|
||||||
import java.net.URISyntaxException;
|
import java.net.URISyntaxException;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
@@ -34,6 +36,9 @@ import java.util.Optional;
|
|||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
import static org.junit.jupiter.api.Assertions.assertThrows;
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
import static org.mockito.ArgumentMatchers.any;
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.Mockito.doNothing;
|
||||||
|
import static org.mockito.Mockito.doThrow;
|
||||||
|
import static org.mockito.Mockito.lenient;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
@@ -43,9 +48,6 @@ class AvailablePluginResourceTest {
|
|||||||
|
|
||||||
private Dispatcher dispatcher;
|
private Dispatcher dispatcher;
|
||||||
|
|
||||||
@Mock
|
|
||||||
Provider<InstalledPluginResource> installedPluginResourceProvider;
|
|
||||||
|
|
||||||
@Mock
|
@Mock
|
||||||
Provider<AvailablePluginResource> availablePluginResourceProvider;
|
Provider<AvailablePluginResource> availablePluginResourceProvider;
|
||||||
|
|
||||||
@@ -63,13 +65,14 @@ class AvailablePluginResourceTest {
|
|||||||
|
|
||||||
PluginRootResource pluginRootResource;
|
PluginRootResource pluginRootResource;
|
||||||
|
|
||||||
private final Subject subject = mock(Subject.class);
|
@Mock
|
||||||
|
Subject subject;
|
||||||
|
|
||||||
|
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
void prepareEnvironment() {
|
void prepareEnvironment() {
|
||||||
dispatcher = MockDispatcherFactory.createDispatcher();
|
dispatcher = MockDispatcherFactory.createDispatcher();
|
||||||
pluginRootResource = new PluginRootResource(installedPluginResourceProvider, availablePluginResourceProvider);
|
pluginRootResource = new PluginRootResource(null, availablePluginResourceProvider, null);
|
||||||
when(availablePluginResourceProvider.get()).thenReturn(availablePluginResource);
|
when(availablePluginResourceProvider.get()).thenReturn(availablePluginResource);
|
||||||
dispatcher.getRegistry().addSingletonResource(pluginRootResource);
|
dispatcher.getRegistry().addSingletonResource(pluginRootResource);
|
||||||
}
|
}
|
||||||
@@ -80,7 +83,7 @@ class AvailablePluginResourceTest {
|
|||||||
@BeforeEach
|
@BeforeEach
|
||||||
void bindSubject() {
|
void bindSubject() {
|
||||||
ThreadContext.bind(subject);
|
ThreadContext.bind(subject);
|
||||||
when(subject.isPermitted(any(String.class))).thenReturn(true);
|
doNothing().when(subject).checkPermission(any(String.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
@AfterEach
|
@AfterEach
|
||||||
@@ -90,9 +93,10 @@ class AvailablePluginResourceTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
void getAvailablePlugins() throws URISyntaxException, UnsupportedEncodingException {
|
void getAvailablePlugins() throws URISyntaxException, UnsupportedEncodingException {
|
||||||
AvailablePlugin plugin = createPlugin();
|
AvailablePlugin plugin = createAvailablePlugin();
|
||||||
|
|
||||||
when(pluginManager.getAvailable()).thenReturn(Collections.singletonList(plugin));
|
when(pluginManager.getAvailable()).thenReturn(Collections.singletonList(plugin));
|
||||||
|
when(pluginManager.getInstalled()).thenReturn(Collections.emptyList());
|
||||||
when(collectionMapper.mapAvailable(Collections.singletonList(plugin))).thenReturn(new MockedResultDto());
|
when(collectionMapper.mapAvailable(Collections.singletonList(plugin))).thenReturn(new MockedResultDto());
|
||||||
|
|
||||||
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/available");
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/available");
|
||||||
@@ -105,13 +109,32 @@ class AvailablePluginResourceTest {
|
|||||||
assertThat(response.getContentAsString()).contains("\"marker\":\"x\"");
|
assertThat(response.getContentAsString()).contains("\"marker\":\"x\"");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotReturnInstalledPlugins() throws URISyntaxException, UnsupportedEncodingException {
|
||||||
|
AvailablePlugin availablePlugin = createAvailablePlugin();
|
||||||
|
InstalledPlugin installedPlugin = createInstalledPlugin();
|
||||||
|
|
||||||
|
when(pluginManager.getAvailable()).thenReturn(Collections.singletonList(availablePlugin));
|
||||||
|
when(pluginManager.getInstalled()).thenReturn(Collections.singletonList(installedPlugin));
|
||||||
|
lenient().when(collectionMapper.mapAvailable(Collections.singletonList(availablePlugin))).thenReturn(new MockedResultDto());
|
||||||
|
|
||||||
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/available");
|
||||||
|
request.accept(VndMediaType.PLUGIN_COLLECTION);
|
||||||
|
MockHttpResponse response = new MockHttpResponse();
|
||||||
|
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(HttpServletResponse.SC_OK).isEqualTo(response.getStatus());
|
||||||
|
assertThat(response.getContentAsString()).doesNotContain("\"marker\":\"x\"");
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void getAvailablePlugin() throws UnsupportedEncodingException, URISyntaxException {
|
void getAvailablePlugin() throws UnsupportedEncodingException, URISyntaxException {
|
||||||
PluginInformation pluginInformation = new PluginInformation();
|
PluginInformation pluginInformation = new PluginInformation();
|
||||||
pluginInformation.setName("pluginName");
|
pluginInformation.setName("pluginName");
|
||||||
pluginInformation.setVersion("2.0.0");
|
pluginInformation.setVersion("2.0.0");
|
||||||
|
|
||||||
AvailablePlugin plugin = createPlugin(pluginInformation);
|
AvailablePlugin plugin = createAvailablePlugin(pluginInformation);
|
||||||
|
|
||||||
when(pluginManager.getAvailable("pluginName")).thenReturn(Optional.of(plugin));
|
when(pluginManager.getAvailable("pluginName")).thenReturn(Optional.of(plugin));
|
||||||
|
|
||||||
@@ -139,38 +162,46 @@ class AvailablePluginResourceTest {
|
|||||||
verify(pluginManager).install("pluginName", false);
|
verify(pluginManager).install("pluginName", false);
|
||||||
assertThat(HttpServletResponse.SC_OK).isEqualTo(response.getStatus());
|
assertThat(HttpServletResponse.SC_OK).isEqualTo(response.getStatus());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
|
||||||
void installPendingPlugin() throws URISyntaxException {
|
|
||||||
MockHttpRequest request = MockHttpRequest.post("/v2/plugins/available/install-pending");
|
|
||||||
MockHttpResponse response = new MockHttpResponse();
|
|
||||||
|
|
||||||
dispatcher.invoke(request, response);
|
|
||||||
|
|
||||||
verify(pluginManager).installPendingAndRestart();
|
|
||||||
assertThat(HttpServletResponse.SC_OK).isEqualTo(response.getStatus());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private AvailablePlugin createPlugin() {
|
private AvailablePlugin createAvailablePlugin() {
|
||||||
return createPlugin(new PluginInformation());
|
PluginInformation pluginInformation = new PluginInformation();
|
||||||
|
pluginInformation.setName("scm-some-plugin");
|
||||||
|
return createAvailablePlugin(pluginInformation);
|
||||||
}
|
}
|
||||||
|
|
||||||
private AvailablePlugin createPlugin(PluginInformation pluginInformation) {
|
private AvailablePlugin createAvailablePlugin(PluginInformation pluginInformation) {
|
||||||
AvailablePluginDescriptor descriptor = new AvailablePluginDescriptor(
|
AvailablePluginDescriptor descriptor = new AvailablePluginDescriptor(
|
||||||
pluginInformation, new PluginCondition(), Collections.emptySet(), "https://download.hitchhiker.com", null
|
pluginInformation, new PluginCondition(), Collections.emptySet(), "https://download.hitchhiker.com", null
|
||||||
);
|
);
|
||||||
return new AvailablePlugin(descriptor);
|
return new AvailablePlugin(descriptor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private InstalledPlugin createInstalledPlugin() {
|
||||||
|
PluginInformation pluginInformation = new PluginInformation();
|
||||||
|
pluginInformation.setName("scm-some-plugin");
|
||||||
|
return createInstalledPlugin(pluginInformation);
|
||||||
|
}
|
||||||
|
|
||||||
|
private InstalledPlugin createInstalledPlugin(PluginInformation pluginInformation) {
|
||||||
|
InstalledPluginDescriptor descriptor = mock(InstalledPluginDescriptor.class);
|
||||||
|
lenient().when(descriptor.getInformation()).thenReturn(pluginInformation);
|
||||||
|
return new InstalledPlugin(descriptor, null, null, null, false);
|
||||||
|
}
|
||||||
|
|
||||||
@Nested
|
@Nested
|
||||||
class WithoutAuthorization {
|
class WithoutAuthorization {
|
||||||
|
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
void unbindSubject() {
|
void bindSubject() {
|
||||||
ThreadContext.unbindSubject();
|
ThreadContext.bind(subject);
|
||||||
|
doThrow(new ShiroException()).when(subject).checkPermission(any(String.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
public void unbindSubject() {
|
||||||
|
ThreadContext.unbindSubject();
|
||||||
|
}
|
||||||
@Test
|
@Test
|
||||||
void shouldNotGetAvailablePluginsIfMissingPermission() throws URISyntaxException {
|
void shouldNotGetAvailablePluginsIfMissingPermission() throws URISyntaxException {
|
||||||
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/available");
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/available");
|
||||||
@@ -178,6 +209,7 @@ class AvailablePluginResourceTest {
|
|||||||
MockHttpResponse response = new MockHttpResponse();
|
MockHttpResponse response = new MockHttpResponse();
|
||||||
|
|
||||||
assertThrows(UnhandledException.class, () -> dispatcher.invoke(request, response));
|
assertThrows(UnhandledException.class, () -> dispatcher.invoke(request, response));
|
||||||
|
verify(subject).checkPermission(any(String.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -187,16 +219,17 @@ class AvailablePluginResourceTest {
|
|||||||
MockHttpResponse response = new MockHttpResponse();
|
MockHttpResponse response = new MockHttpResponse();
|
||||||
|
|
||||||
assertThrows(UnhandledException.class, () -> dispatcher.invoke(request, response));
|
assertThrows(UnhandledException.class, () -> dispatcher.invoke(request, response));
|
||||||
|
verify(subject).checkPermission(any(String.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldNotInstallPluginIfMissingPermission() throws URISyntaxException {
|
void shouldNotInstallPluginIfMissingPermission() throws URISyntaxException {
|
||||||
ThreadContext.unbindSubject();
|
|
||||||
MockHttpRequest request = MockHttpRequest.post("/v2/plugins/available/pluginName/install");
|
MockHttpRequest request = MockHttpRequest.post("/v2/plugins/available/pluginName/install");
|
||||||
request.accept(VndMediaType.PLUGIN);
|
request.accept(VndMediaType.PLUGIN);
|
||||||
MockHttpResponse response = new MockHttpResponse();
|
MockHttpResponse response = new MockHttpResponse();
|
||||||
|
|
||||||
assertThrows(UnhandledException.class, () -> dispatcher.invoke(request, response));
|
assertThrows(UnhandledException.class, () -> dispatcher.invoke(request, response));
|
||||||
|
verify(subject).checkPermission(any(String.class));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -29,10 +29,12 @@ import java.net.URISyntaxException;
|
|||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import static java.util.Collections.emptyList;
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
import static org.junit.jupiter.api.Assertions.assertThrows;
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
import static org.mockito.ArgumentMatchers.any;
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
import static org.mockito.Mockito.*;
|
import static org.mockito.Mockito.*;
|
||||||
|
import static sonia.scm.plugin.PluginTestHelper.createInstalled;
|
||||||
|
|
||||||
@ExtendWith(MockitoExtension.class)
|
@ExtendWith(MockitoExtension.class)
|
||||||
class InstalledPluginResourceTest {
|
class InstalledPluginResourceTest {
|
||||||
@@ -64,7 +66,7 @@ class InstalledPluginResourceTest {
|
|||||||
@BeforeEach
|
@BeforeEach
|
||||||
void prepareEnvironment() {
|
void prepareEnvironment() {
|
||||||
dispatcher = MockDispatcherFactory.createDispatcher();
|
dispatcher = MockDispatcherFactory.createDispatcher();
|
||||||
pluginRootResource = new PluginRootResource(installedPluginResourceProvider, availablePluginResourceProvider);
|
pluginRootResource = new PluginRootResource(installedPluginResourceProvider, null, null);
|
||||||
when(installedPluginResourceProvider.get()).thenReturn(installedPluginResource);
|
when(installedPluginResourceProvider.get()).thenReturn(installedPluginResource);
|
||||||
dispatcher.getRegistry().addSingletonResource(pluginRootResource);
|
dispatcher.getRegistry().addSingletonResource(pluginRootResource);
|
||||||
}
|
}
|
||||||
@@ -85,9 +87,9 @@ class InstalledPluginResourceTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
void getInstalledPlugins() throws URISyntaxException, UnsupportedEncodingException {
|
void getInstalledPlugins() throws URISyntaxException, UnsupportedEncodingException {
|
||||||
InstalledPlugin installedPlugin = createPlugin();
|
InstalledPlugin installedPlugin = createInstalled("");
|
||||||
when(pluginManager.getInstalled()).thenReturn(Collections.singletonList(installedPlugin));
|
when(pluginManager.getInstalled()).thenReturn(Collections.singletonList(installedPlugin));
|
||||||
when(collectionMapper.mapInstalled(Collections.singletonList(installedPlugin))).thenReturn(new MockedResultDto());
|
when(collectionMapper.mapInstalled(Collections.singletonList(installedPlugin), Collections.emptyList())).thenReturn(new MockedResultDto());
|
||||||
|
|
||||||
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/installed");
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/installed");
|
||||||
request.accept(VndMediaType.PLUGIN_COLLECTION);
|
request.accept(VndMediaType.PLUGIN_COLLECTION);
|
||||||
@@ -104,13 +106,13 @@ class InstalledPluginResourceTest {
|
|||||||
PluginInformation pluginInformation = new PluginInformation();
|
PluginInformation pluginInformation = new PluginInformation();
|
||||||
pluginInformation.setVersion("2.0.0");
|
pluginInformation.setVersion("2.0.0");
|
||||||
pluginInformation.setName("pluginName");
|
pluginInformation.setName("pluginName");
|
||||||
InstalledPlugin installedPlugin = createPlugin(pluginInformation);
|
InstalledPlugin installedPlugin = createInstalled(pluginInformation);
|
||||||
|
|
||||||
when(pluginManager.getInstalled("pluginName")).thenReturn(Optional.of(installedPlugin));
|
when(pluginManager.getInstalled("pluginName")).thenReturn(Optional.of(installedPlugin));
|
||||||
|
|
||||||
PluginDto pluginDto = new PluginDto();
|
PluginDto pluginDto = new PluginDto();
|
||||||
pluginDto.setName("pluginName");
|
pluginDto.setName("pluginName");
|
||||||
when(mapper.mapInstalled(installedPlugin)).thenReturn(pluginDto);
|
when(mapper.mapInstalled(installedPlugin, emptyList())).thenReturn(pluginDto);
|
||||||
|
|
||||||
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/installed/pluginName");
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/installed/pluginName");
|
||||||
request.accept(VndMediaType.PLUGIN);
|
request.accept(VndMediaType.PLUGIN);
|
||||||
@@ -123,18 +125,6 @@ class InstalledPluginResourceTest {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private InstalledPlugin createPlugin() {
|
|
||||||
return createPlugin(new PluginInformation());
|
|
||||||
}
|
|
||||||
|
|
||||||
private InstalledPlugin createPlugin(PluginInformation information) {
|
|
||||||
InstalledPlugin plugin = mock(InstalledPlugin.class);
|
|
||||||
InstalledPluginDescriptor descriptor = mock(InstalledPluginDescriptor.class);
|
|
||||||
lenient().when(descriptor.getInformation()).thenReturn(information);
|
|
||||||
lenient().when(plugin.getDescriptor()).thenReturn(descriptor);
|
|
||||||
return plugin;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nested
|
@Nested
|
||||||
class WithoutAuthorization {
|
class WithoutAuthorization {
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,241 @@
|
|||||||
|
package sonia.scm.api.v2.resources;
|
||||||
|
|
||||||
|
import com.google.inject.util.Providers;
|
||||||
|
import org.apache.shiro.ShiroException;
|
||||||
|
import org.apache.shiro.subject.Subject;
|
||||||
|
import org.apache.shiro.util.ThreadContext;
|
||||||
|
import org.jboss.resteasy.core.Dispatcher;
|
||||||
|
import org.jboss.resteasy.mock.MockDispatcherFactory;
|
||||||
|
import org.jboss.resteasy.mock.MockHttpRequest;
|
||||||
|
import org.jboss.resteasy.mock.MockHttpResponse;
|
||||||
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Nested;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith;
|
||||||
|
import org.mockito.InjectMocks;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
|
import sonia.scm.plugin.AvailablePlugin;
|
||||||
|
import sonia.scm.plugin.AvailablePluginDescriptor;
|
||||||
|
import sonia.scm.plugin.InstalledPlugin;
|
||||||
|
import sonia.scm.plugin.InstalledPluginDescriptor;
|
||||||
|
import sonia.scm.plugin.PluginInformation;
|
||||||
|
import sonia.scm.plugin.PluginManager;
|
||||||
|
|
||||||
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
import javax.ws.rs.core.Response;
|
||||||
|
import javax.ws.rs.ext.ExceptionMapper;
|
||||||
|
import java.io.UnsupportedEncodingException;
|
||||||
|
import java.net.URISyntaxException;
|
||||||
|
|
||||||
|
import static java.net.URI.create;
|
||||||
|
import static java.util.Collections.emptyList;
|
||||||
|
import static java.util.Collections.singletonList;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.Mockito.doNothing;
|
||||||
|
import static org.mockito.Mockito.doThrow;
|
||||||
|
import static org.mockito.Mockito.lenient;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.never;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
@ExtendWith(MockitoExtension.class)
|
||||||
|
class PendingPluginResourceTest {
|
||||||
|
|
||||||
|
Dispatcher dispatcher = MockDispatcherFactory.createDispatcher();
|
||||||
|
|
||||||
|
ResourceLinks resourceLinks = ResourceLinksMock.createMock(create("/"));
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
PluginManager pluginManager;
|
||||||
|
@Mock
|
||||||
|
PluginDtoMapper mapper;
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
Subject subject;
|
||||||
|
|
||||||
|
@InjectMocks
|
||||||
|
PendingPluginResource pendingPluginResource;
|
||||||
|
|
||||||
|
MockHttpResponse response = new MockHttpResponse();
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void prepareEnvironment() {
|
||||||
|
dispatcher = MockDispatcherFactory.createDispatcher();
|
||||||
|
dispatcher.getProviderFactory().register(new PermissionExceptionMapper());
|
||||||
|
PluginRootResource pluginRootResource = new PluginRootResource(null, null, Providers.of(pendingPluginResource));
|
||||||
|
dispatcher.getRegistry().addSingletonResource(pluginRootResource);
|
||||||
|
}
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void mockMapper() {
|
||||||
|
lenient().when(mapper.mapAvailable(any())).thenAnswer(invocation -> {
|
||||||
|
PluginDto dto = new PluginDto();
|
||||||
|
dto.setName(((AvailablePlugin)invocation.getArgument(0)).getDescriptor().getInformation().getName());
|
||||||
|
return dto;
|
||||||
|
});
|
||||||
|
lenient().when(mapper.mapInstalled(any(), any())).thenAnswer(invocation -> {
|
||||||
|
PluginDto dto = new PluginDto();
|
||||||
|
dto.setName(((InstalledPlugin)invocation.getArgument(0)).getDescriptor().getInformation().getName());
|
||||||
|
return dto;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class withAuthorization {
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void bindSubject() {
|
||||||
|
ThreadContext.bind(subject);
|
||||||
|
doNothing().when(subject).checkPermission("plugin:manage");
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
void unbindSubject() {
|
||||||
|
ThreadContext.unbindSubject();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldGetEmptyPluginListsWithoutInstallLinkWhenNoPendingPluginsPresent() throws URISyntaxException, UnsupportedEncodingException {
|
||||||
|
AvailablePlugin availablePlugin = createAvailablePlugin("not-pending-plugin");
|
||||||
|
when(availablePlugin.isPending()).thenReturn(false);
|
||||||
|
when(pluginManager.getAvailable()).thenReturn(singletonList(availablePlugin));
|
||||||
|
|
||||||
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/pending");
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(response.getStatus()).isEqualTo(HttpServletResponse.SC_OK);
|
||||||
|
assertThat(response.getContentAsString()).contains("\"_links\":{\"self\":{\"href\":\"/v2/plugins/pending\"}}");
|
||||||
|
assertThat(response.getContentAsString()).doesNotContain("not-pending-plugin");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldGetPendingAvailablePluginListWithInstallLink() throws URISyntaxException, UnsupportedEncodingException {
|
||||||
|
AvailablePlugin availablePlugin = createAvailablePlugin("pending-available-plugin");
|
||||||
|
when(availablePlugin.isPending()).thenReturn(true);
|
||||||
|
when(pluginManager.getAvailable()).thenReturn(singletonList(availablePlugin));
|
||||||
|
|
||||||
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/pending");
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(response.getStatus()).isEqualTo(HttpServletResponse.SC_OK);
|
||||||
|
assertThat(response.getContentAsString()).contains("\"new\":[{\"name\":\"pending-available-plugin\"");
|
||||||
|
assertThat(response.getContentAsString()).contains("\"execute\":{\"href\":\"/v2/plugins/pending/execute\"}");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldGetPendingUpdatePluginListWithInstallLink() throws URISyntaxException, UnsupportedEncodingException {
|
||||||
|
AvailablePlugin availablePlugin = createAvailablePlugin("available-plugin");
|
||||||
|
when(availablePlugin.isPending()).thenReturn(true);
|
||||||
|
when(pluginManager.getAvailable()).thenReturn(singletonList(availablePlugin));
|
||||||
|
InstalledPlugin installedPlugin = createInstalledPlugin("available-plugin");
|
||||||
|
when(pluginManager.getInstalled()).thenReturn(singletonList(installedPlugin));
|
||||||
|
|
||||||
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/pending");
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(response.getStatus()).isEqualTo(HttpServletResponse.SC_OK);
|
||||||
|
assertThat(response.getContentAsString()).contains("\"update\":[{\"name\":\"available-plugin\"");
|
||||||
|
assertThat(response.getContentAsString()).contains("\"execute\":{\"href\":\"/v2/plugins/pending/execute\"}");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldGetPendingUninstallPluginListWithInstallLink() throws URISyntaxException, UnsupportedEncodingException {
|
||||||
|
when(pluginManager.getAvailable()).thenReturn(emptyList());
|
||||||
|
InstalledPlugin installedPlugin = createInstalledPlugin("uninstalled-plugin");
|
||||||
|
when(installedPlugin.isMarkedForUninstall()).thenReturn(true);
|
||||||
|
when(pluginManager.getInstalled()).thenReturn(singletonList(installedPlugin));
|
||||||
|
|
||||||
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/pending");
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(response.getStatus()).isEqualTo(HttpServletResponse.SC_OK);
|
||||||
|
assertThat(response.getContentAsString()).contains("\"uninstall\":[{\"name\":\"uninstalled-plugin\"");
|
||||||
|
assertThat(response.getContentAsString()).contains("\"execute\":{\"href\":\"/v2/plugins/pending/execute\"}");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldExecutePendingPlugins() throws URISyntaxException {
|
||||||
|
MockHttpRequest request = MockHttpRequest.post("/v2/plugins/pending/execute");
|
||||||
|
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(response.getStatus()).isEqualTo(HttpServletResponse.SC_OK);
|
||||||
|
verify(pluginManager).executePendingAndRestart();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
class WithoutAuthorization {
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void bindSubject() {
|
||||||
|
ThreadContext.bind(subject);
|
||||||
|
doThrow(new ShiroException()).when(subject).checkPermission("plugin:manage");
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
void unbindSubject() {
|
||||||
|
ThreadContext.unbindSubject();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotListPendingPlugins() throws URISyntaxException {
|
||||||
|
MockHttpRequest request = MockHttpRequest.get("/v2/plugins/pending");
|
||||||
|
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(response.getStatus()).isEqualTo(HttpServletResponse.SC_UNAUTHORIZED);
|
||||||
|
verify(pluginManager, never()).executePendingAndRestart();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotExecutePendingPlugins() throws URISyntaxException {
|
||||||
|
MockHttpRequest request = MockHttpRequest.post("/v2/plugins/pending/execute");
|
||||||
|
|
||||||
|
dispatcher.invoke(request, response);
|
||||||
|
|
||||||
|
assertThat(response.getStatus()).isEqualTo(HttpServletResponse.SC_UNAUTHORIZED);
|
||||||
|
verify(pluginManager, never()).executePendingAndRestart();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static class PermissionExceptionMapper implements ExceptionMapper<ShiroException> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Response toResponse(ShiroException exception) {
|
||||||
|
return Response.status(401).entity(exception.getMessage()).build();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private AvailablePlugin createAvailablePlugin(String name) {
|
||||||
|
PluginInformation pluginInformation = new PluginInformation();
|
||||||
|
pluginInformation.setName(name);
|
||||||
|
return createAvailablePlugin(pluginInformation);
|
||||||
|
}
|
||||||
|
|
||||||
|
private AvailablePlugin createAvailablePlugin(PluginInformation pluginInformation) {
|
||||||
|
AvailablePluginDescriptor descriptor = mock(AvailablePluginDescriptor.class);
|
||||||
|
lenient().when(descriptor.getInformation()).thenReturn(pluginInformation);
|
||||||
|
AvailablePlugin availablePlugin = mock(AvailablePlugin.class);
|
||||||
|
lenient().when(availablePlugin.getDescriptor()).thenReturn(descriptor);
|
||||||
|
return availablePlugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
private InstalledPlugin createInstalledPlugin(String name) {
|
||||||
|
PluginInformation pluginInformation = new PluginInformation();
|
||||||
|
pluginInformation.setName(name);
|
||||||
|
return createInstalledPlugin(pluginInformation);
|
||||||
|
}
|
||||||
|
|
||||||
|
private InstalledPlugin createInstalledPlugin(PluginInformation pluginInformation) {
|
||||||
|
InstalledPluginDescriptor descriptor = mock(InstalledPluginDescriptor.class);
|
||||||
|
lenient().when(descriptor.getInformation()).thenReturn(pluginInformation);
|
||||||
|
InstalledPlugin installedPlugin = mock(InstalledPlugin.class);
|
||||||
|
lenient().when(installedPlugin.getDescriptor()).thenReturn(descriptor);
|
||||||
|
return installedPlugin;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,171 @@
|
|||||||
|
package sonia.scm.api.v2.resources;
|
||||||
|
|
||||||
|
import de.otto.edison.hal.HalRepresentation;
|
||||||
|
import org.apache.shiro.subject.Subject;
|
||||||
|
import org.apache.shiro.subject.support.SubjectThreadState;
|
||||||
|
import org.apache.shiro.util.ThreadContext;
|
||||||
|
import org.apache.shiro.util.ThreadState;
|
||||||
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith;
|
||||||
|
import org.mockito.InjectMocks;
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
|
import sonia.scm.plugin.AvailablePlugin;
|
||||||
|
import sonia.scm.plugin.AvailablePluginDescriptor;
|
||||||
|
import sonia.scm.plugin.InstalledPlugin;
|
||||||
|
import sonia.scm.plugin.InstalledPluginDescriptor;
|
||||||
|
import sonia.scm.plugin.PluginInformation;
|
||||||
|
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import static java.util.Collections.singletonList;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.mockito.Mockito.lenient;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
@ExtendWith(MockitoExtension.class)
|
||||||
|
class PluginDtoCollectionMapperTest {
|
||||||
|
|
||||||
|
ResourceLinks resourceLinks = ResourceLinksMock.createMock(URI.create("/"));
|
||||||
|
|
||||||
|
@InjectMocks
|
||||||
|
PluginDtoMapperImpl pluginDtoMapper;
|
||||||
|
|
||||||
|
Subject subject = mock(Subject.class);
|
||||||
|
ThreadState subjectThreadState = new SubjectThreadState(subject);
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void bindSubject() {
|
||||||
|
subjectThreadState.bind();
|
||||||
|
ThreadContext.bind(subject);
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
public void unbindSubject() {
|
||||||
|
ThreadContext.unbindSubject();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldMapInstalledPluginsWithoutUpdateWhenNoNewerVersionIsAvailable() {
|
||||||
|
PluginDtoCollectionMapper mapper = new PluginDtoCollectionMapper(resourceLinks, pluginDtoMapper);
|
||||||
|
|
||||||
|
HalRepresentation result = mapper.mapInstalled(
|
||||||
|
singletonList(createInstalledPlugin("scm-some-plugin", "1")),
|
||||||
|
singletonList(createAvailablePlugin("scm-other-plugin", "2")));
|
||||||
|
|
||||||
|
List<HalRepresentation> plugins = result.getEmbedded().getItemsBy("plugins");
|
||||||
|
assertThat(plugins).hasSize(1);
|
||||||
|
PluginDto plugin = (PluginDto) plugins.get(0);
|
||||||
|
assertThat(plugin.getVersion()).isEqualTo("1");
|
||||||
|
assertThat(plugin.getNewVersion()).isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldSetNewVersionInInstalledPluginWhenAvailableVersionIsNewer() {
|
||||||
|
PluginDtoCollectionMapper mapper = new PluginDtoCollectionMapper(resourceLinks, pluginDtoMapper);
|
||||||
|
|
||||||
|
HalRepresentation result = mapper.mapInstalled(
|
||||||
|
singletonList(createInstalledPlugin("scm-some-plugin", "1")),
|
||||||
|
singletonList(createAvailablePlugin("scm-some-plugin", "2")));
|
||||||
|
|
||||||
|
PluginDto plugin = getPluginDtoFromResult(result);
|
||||||
|
assertThat(plugin.getVersion()).isEqualTo("1");
|
||||||
|
assertThat(plugin.getNewVersion()).isEqualTo("2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotAddInstallLinkForNewVersionWhenNotPermitted() {
|
||||||
|
when(subject.isPermitted("plugin:manage")).thenReturn(false);
|
||||||
|
PluginDtoCollectionMapper mapper = new PluginDtoCollectionMapper(resourceLinks, pluginDtoMapper);
|
||||||
|
|
||||||
|
HalRepresentation result = mapper.mapInstalled(
|
||||||
|
singletonList(createInstalledPlugin("scm-some-plugin", "1")),
|
||||||
|
singletonList(createAvailablePlugin("scm-some-plugin", "2")));
|
||||||
|
|
||||||
|
PluginDto plugin = getPluginDtoFromResult(result);
|
||||||
|
assertThat(plugin.getLinks().getLinkBy("update")).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotAddInstallLinkForNewVersionWhenInstallationIsPending() {
|
||||||
|
when(subject.isPermitted("plugin:manage")).thenReturn(true);
|
||||||
|
PluginDtoCollectionMapper mapper = new PluginDtoCollectionMapper(resourceLinks, pluginDtoMapper);
|
||||||
|
|
||||||
|
AvailablePlugin availablePlugin = createAvailablePlugin("scm-some-plugin", "2");
|
||||||
|
when(availablePlugin.isPending()).thenReturn(true);
|
||||||
|
HalRepresentation result = mapper.mapInstalled(
|
||||||
|
singletonList(createInstalledPlugin("scm-some-plugin", "1")),
|
||||||
|
singletonList(availablePlugin));
|
||||||
|
|
||||||
|
PluginDto plugin = getPluginDtoFromResult(result);
|
||||||
|
assertThat(plugin.getLinks().getLinkBy("update")).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldAddInstallLinkForNewVersionWhenPermitted() {
|
||||||
|
when(subject.isPermitted("plugin:manage")).thenReturn(true);
|
||||||
|
PluginDtoCollectionMapper mapper = new PluginDtoCollectionMapper(resourceLinks, pluginDtoMapper);
|
||||||
|
|
||||||
|
HalRepresentation result = mapper.mapInstalled(
|
||||||
|
singletonList(createInstalledPlugin("scm-some-plugin", "1")),
|
||||||
|
singletonList(createAvailablePlugin("scm-some-plugin", "2")));
|
||||||
|
|
||||||
|
PluginDto plugin = getPluginDtoFromResult(result);
|
||||||
|
assertThat(plugin.getLinks().getLinkBy("update")).isNotEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldSetInstalledPluginPendingWhenCorrespondingAvailablePluginIsPending() {
|
||||||
|
when(subject.isPermitted("plugin:manage")).thenReturn(true);
|
||||||
|
PluginDtoCollectionMapper mapper = new PluginDtoCollectionMapper(resourceLinks, pluginDtoMapper);
|
||||||
|
|
||||||
|
AvailablePlugin availablePlugin = createAvailablePlugin("scm-some-plugin", "2");
|
||||||
|
when(availablePlugin.isPending()).thenReturn(true);
|
||||||
|
HalRepresentation result = mapper.mapInstalled(
|
||||||
|
singletonList(createInstalledPlugin("scm-some-plugin", "1")),
|
||||||
|
singletonList(availablePlugin));
|
||||||
|
|
||||||
|
PluginDto plugin = getPluginDtoFromResult(result);
|
||||||
|
assertThat(plugin.isPending()).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
private PluginDto getPluginDtoFromResult(HalRepresentation result) {
|
||||||
|
assertThat(result.getEmbedded().getItemsBy("plugins")).hasSize(1);
|
||||||
|
List<HalRepresentation> plugins = result.getEmbedded().getItemsBy("plugins");
|
||||||
|
return (PluginDto) plugins.get(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
private InstalledPlugin createInstalledPlugin(String name, String version) {
|
||||||
|
PluginInformation information = new PluginInformation();
|
||||||
|
information.setName(name);
|
||||||
|
information.setVersion(version);
|
||||||
|
return createInstalledPlugin(information);
|
||||||
|
}
|
||||||
|
|
||||||
|
private InstalledPlugin createInstalledPlugin(PluginInformation information) {
|
||||||
|
InstalledPlugin plugin = mock(InstalledPlugin.class);
|
||||||
|
InstalledPluginDescriptor descriptor = mock(InstalledPluginDescriptor.class);
|
||||||
|
lenient().when(descriptor.getInformation()).thenReturn(information);
|
||||||
|
lenient().when(plugin.getDescriptor()).thenReturn(descriptor);
|
||||||
|
return plugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
private AvailablePlugin createAvailablePlugin(String name, String version) {
|
||||||
|
PluginInformation information = new PluginInformation();
|
||||||
|
information.setName(name);
|
||||||
|
information.setVersion(version);
|
||||||
|
return createAvailablePlugin(information);
|
||||||
|
}
|
||||||
|
|
||||||
|
private AvailablePlugin createAvailablePlugin(PluginInformation information) {
|
||||||
|
AvailablePlugin plugin = mock(AvailablePlugin.class);
|
||||||
|
AvailablePluginDescriptor descriptor = mock(AvailablePluginDescriptor.class);
|
||||||
|
lenient().when(descriptor.getInformation()).thenReturn(information);
|
||||||
|
lenient().when(plugin.getDescriptor()).thenReturn(descriptor);
|
||||||
|
return plugin;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -17,10 +17,14 @@ import sonia.scm.plugin.InstalledPlugin;
|
|||||||
import sonia.scm.plugin.PluginInformation;
|
import sonia.scm.plugin.PluginInformation;
|
||||||
|
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
|
import java.util.Collections;
|
||||||
|
|
||||||
|
import static java.util.Collections.emptyList;
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
import static sonia.scm.plugin.PluginTestHelper.createAvailable;
|
||||||
|
import static sonia.scm.plugin.PluginTestHelper.createInstalled;
|
||||||
|
|
||||||
@ExtendWith(MockitoExtension.class)
|
@ExtendWith(MockitoExtension.class)
|
||||||
class PluginDtoMapperTest {
|
class PluginDtoMapperTest {
|
||||||
@@ -72,22 +76,16 @@ class PluginDtoMapperTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldAppendInstalledSelfLink() {
|
void shouldAppendInstalledSelfLink() {
|
||||||
InstalledPlugin plugin = createInstalled();
|
InstalledPlugin plugin = createInstalled(createPluginInformation());
|
||||||
|
|
||||||
PluginDto dto = mapper.mapInstalled(plugin);
|
PluginDto dto = mapper.mapInstalled(plugin, emptyList());
|
||||||
assertThat(dto.getLinks().getLinkBy("self").get().getHref())
|
assertThat(dto.getLinks().getLinkBy("self").get().getHref())
|
||||||
.isEqualTo("https://hitchhiker.com/v2/plugins/installed/scm-cas-plugin");
|
.isEqualTo("https://hitchhiker.com/v2/plugins/installed/scm-cas-plugin");
|
||||||
}
|
}
|
||||||
|
|
||||||
private InstalledPlugin createInstalled(PluginInformation information) {
|
|
||||||
InstalledPlugin plugin = mock(InstalledPlugin.class, Answers.RETURNS_DEEP_STUBS);
|
|
||||||
when(plugin.getDescriptor().getInformation()).thenReturn(information);
|
|
||||||
return plugin;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldAppendAvailableSelfLink() {
|
void shouldAppendAvailableSelfLink() {
|
||||||
AvailablePlugin plugin = createAvailable();
|
AvailablePlugin plugin = createAvailable(createPluginInformation());
|
||||||
|
|
||||||
PluginDto dto = mapper.mapAvailable(plugin);
|
PluginDto dto = mapper.mapAvailable(plugin);
|
||||||
assertThat(dto.getLinks().getLinkBy("self").get().getHref())
|
assertThat(dto.getLinks().getLinkBy("self").get().getHref())
|
||||||
@@ -96,7 +94,7 @@ class PluginDtoMapperTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldNotAppendInstallLinkWithoutPermissions() {
|
void shouldNotAppendInstallLinkWithoutPermissions() {
|
||||||
AvailablePlugin plugin = createAvailable();
|
AvailablePlugin plugin = createAvailable(createPluginInformation());
|
||||||
|
|
||||||
PluginDto dto = mapper.mapAvailable(plugin);
|
PluginDto dto = mapper.mapAvailable(plugin);
|
||||||
assertThat(dto.getLinks().getLinkBy("install")).isEmpty();
|
assertThat(dto.getLinks().getLinkBy("install")).isEmpty();
|
||||||
@@ -105,7 +103,7 @@ class PluginDtoMapperTest {
|
|||||||
@Test
|
@Test
|
||||||
void shouldAppendInstallLink() {
|
void shouldAppendInstallLink() {
|
||||||
when(subject.isPermitted("plugin:manage")).thenReturn(true);
|
when(subject.isPermitted("plugin:manage")).thenReturn(true);
|
||||||
AvailablePlugin plugin = createAvailable();
|
AvailablePlugin plugin = createAvailable(createPluginInformation());
|
||||||
|
|
||||||
PluginDto dto = mapper.mapAvailable(plugin);
|
PluginDto dto = mapper.mapAvailable(plugin);
|
||||||
assertThat(dto.getLinks().getLinkBy("install").get().getHref())
|
assertThat(dto.getLinks().getLinkBy("install").get().getHref())
|
||||||
@@ -123,25 +121,21 @@ class PluginDtoMapperTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldAppendDependencies() {
|
void shouldAppendDependencies() {
|
||||||
AvailablePlugin plugin = createAvailable();
|
AvailablePlugin plugin = createAvailable(createPluginInformation());
|
||||||
when(plugin.getDescriptor().getDependencies()).thenReturn(ImmutableSet.of("one", "two"));
|
when(plugin.getDescriptor().getDependencies()).thenReturn(ImmutableSet.of("one", "two"));
|
||||||
|
|
||||||
PluginDto dto = mapper.mapAvailable(plugin);
|
PluginDto dto = mapper.mapAvailable(plugin);
|
||||||
assertThat(dto.getDependencies()).containsOnly("one", "two");
|
assertThat(dto.getDependencies()).containsOnly("one", "two");
|
||||||
}
|
}
|
||||||
|
|
||||||
private InstalledPlugin createInstalled() {
|
@Test
|
||||||
return createInstalled(createPluginInformation());
|
void shouldAppendUninstallLink() {
|
||||||
}
|
when(subject.isPermitted("plugin:manage")).thenReturn(true);
|
||||||
|
InstalledPlugin plugin = createInstalled(createPluginInformation());
|
||||||
|
when(plugin.isUninstallable()).thenReturn(true);
|
||||||
|
|
||||||
private AvailablePlugin createAvailable() {
|
PluginDto dto = mapper.mapInstalled(plugin, emptyList());
|
||||||
return createAvailable(createPluginInformation());
|
assertThat(dto.getLinks().getLinkBy("uninstall").get().getHref())
|
||||||
|
.isEqualTo("https://hitchhiker.com/v2/plugins/installed/scm-cas-plugin/uninstall");
|
||||||
}
|
}
|
||||||
|
|
||||||
private AvailablePlugin createAvailable(PluginInformation information) {
|
|
||||||
AvailablePluginDescriptor descriptor = mock(AvailablePluginDescriptor.class);
|
|
||||||
when(descriptor.getInformation()).thenReturn(information);
|
|
||||||
return new AvailablePlugin(descriptor);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -38,6 +38,7 @@ public class ResourceLinksMock {
|
|||||||
when(resourceLinks.repositoryTypeCollection()).thenReturn(new ResourceLinks.RepositoryTypeCollectionLinks(uriInfo));
|
when(resourceLinks.repositoryTypeCollection()).thenReturn(new ResourceLinks.RepositoryTypeCollectionLinks(uriInfo));
|
||||||
when(resourceLinks.installedPluginCollection()).thenReturn(new ResourceLinks.InstalledPluginCollectionLinks(uriInfo));
|
when(resourceLinks.installedPluginCollection()).thenReturn(new ResourceLinks.InstalledPluginCollectionLinks(uriInfo));
|
||||||
when(resourceLinks.availablePluginCollection()).thenReturn(new ResourceLinks.AvailablePluginCollectionLinks(uriInfo));
|
when(resourceLinks.availablePluginCollection()).thenReturn(new ResourceLinks.AvailablePluginCollectionLinks(uriInfo));
|
||||||
|
when(resourceLinks.pendingPluginCollection()).thenReturn(new ResourceLinks.PendingPluginCollectionLinks(uriInfo));
|
||||||
when(resourceLinks.installedPlugin()).thenReturn(new ResourceLinks.InstalledPluginLinks(uriInfo));
|
when(resourceLinks.installedPlugin()).thenReturn(new ResourceLinks.InstalledPluginLinks(uriInfo));
|
||||||
when(resourceLinks.availablePlugin()).thenReturn(new ResourceLinks.AvailablePluginLinks(uriInfo));
|
when(resourceLinks.availablePlugin()).thenReturn(new ResourceLinks.AvailablePluginLinks(uriInfo));
|
||||||
when(resourceLinks.uiPluginCollection()).thenReturn(new ResourceLinks.UIPluginCollectionLinks(uriInfo));
|
when(resourceLinks.uiPluginCollection()).thenReturn(new ResourceLinks.UIPluginCollectionLinks(uriInfo));
|
||||||
|
|||||||
@@ -10,24 +10,38 @@ import org.junit.jupiter.api.BeforeEach;
|
|||||||
import org.junit.jupiter.api.Nested;
|
import org.junit.jupiter.api.Nested;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.api.extension.ExtendWith;
|
import org.junit.jupiter.api.extension.ExtendWith;
|
||||||
import org.mockito.Answers;
|
import org.junitpioneer.jupiter.TempDirectory;
|
||||||
import org.mockito.ArgumentCaptor;
|
import org.mockito.ArgumentCaptor;
|
||||||
import org.mockito.InjectMocks;
|
import org.mockito.InjectMocks;
|
||||||
import org.mockito.Mock;
|
import org.mockito.Mock;
|
||||||
import org.mockito.junit.jupiter.MockitoExtension;
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
import sonia.scm.NotFoundException;
|
import sonia.scm.NotFoundException;
|
||||||
|
import sonia.scm.ScmConstraintViolationException;
|
||||||
import sonia.scm.event.ScmEventBus;
|
import sonia.scm.event.ScmEventBus;
|
||||||
import sonia.scm.lifecycle.RestartEvent;
|
import sonia.scm.lifecycle.RestartEvent;
|
||||||
|
|
||||||
|
import java.nio.file.Path;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import static java.util.Arrays.asList;
|
||||||
|
import static java.util.Collections.singleton;
|
||||||
|
import static java.util.Collections.singletonList;
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
import static org.assertj.core.api.Assertions.in;
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertThrows;
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
import static org.mockito.Mockito.*;
|
import static org.mockito.Mockito.any;
|
||||||
|
import static org.mockito.Mockito.doReturn;
|
||||||
|
import static org.mockito.Mockito.doThrow;
|
||||||
|
import static org.mockito.Mockito.lenient;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.never;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
import static sonia.scm.plugin.PluginTestHelper.createAvailable;
|
||||||
|
import static sonia.scm.plugin.PluginTestHelper.createInstalled;
|
||||||
|
|
||||||
@ExtendWith(MockitoExtension.class)
|
@ExtendWith(MockitoExtension.class)
|
||||||
|
@ExtendWith(TempDirectory.class)
|
||||||
class DefaultPluginManagerTest {
|
class DefaultPluginManagerTest {
|
||||||
|
|
||||||
@Mock
|
@Mock
|
||||||
@@ -269,14 +283,14 @@ class DefaultPluginManagerTest {
|
|||||||
when(center.getAvailable()).thenReturn(ImmutableSet.of(review));
|
when(center.getAvailable()).thenReturn(ImmutableSet.of(review));
|
||||||
|
|
||||||
manager.install("scm-review-plugin", false);
|
manager.install("scm-review-plugin", false);
|
||||||
manager.installPendingAndRestart();
|
manager.executePendingAndRestart();
|
||||||
|
|
||||||
verify(eventBus).post(any(RestartEvent.class));
|
verify(eventBus).post(any(RestartEvent.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldNotSendRestartEventWithoutPendingPlugins() {
|
void shouldNotSendRestartEventWithoutPendingPlugins() {
|
||||||
manager.installPendingAndRestart();
|
manager.executePendingAndRestart();
|
||||||
|
|
||||||
verify(eventBus, never()).post(any());
|
verify(eventBus, never()).post(any());
|
||||||
}
|
}
|
||||||
@@ -303,6 +317,116 @@ class DefaultPluginManagerTest {
|
|||||||
assertThat(available.get(0).isPending()).isTrue();
|
assertThat(available.get(0).isPending()).isTrue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldThrowExceptionWhenUninstallingUnknownPlugin() {
|
||||||
|
assertThrows(NotFoundException.class, () -> manager.uninstall("no-such-plugin", false));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldUseDependencyTrackerForUninstall() {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
InstalledPlugin reviewPlugin = createInstalled("scm-review-plugin");
|
||||||
|
when(reviewPlugin.getDescriptor().getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(ImmutableList.of(mailPlugin, reviewPlugin));
|
||||||
|
manager.computeInstallationDependencies();
|
||||||
|
|
||||||
|
assertThrows(ScmConstraintViolationException.class, () -> manager.uninstall("scm-mail-plugin", false));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldCreateUninstallFile(@TempDirectory.TempDir Path temp) {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
when(mailPlugin.getDirectory()).thenReturn(temp);
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(singletonList(mailPlugin));
|
||||||
|
|
||||||
|
manager.uninstall("scm-mail-plugin", false);
|
||||||
|
|
||||||
|
assertThat(temp.resolve("uninstall")).exists();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldMarkPluginForUninstall(@TempDirectory.TempDir Path temp) {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
when(mailPlugin.getDirectory()).thenReturn(temp);
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(singletonList(mailPlugin));
|
||||||
|
|
||||||
|
manager.uninstall("scm-mail-plugin", false);
|
||||||
|
|
||||||
|
verify(mailPlugin).setMarkedForUninstall(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldThrowExceptionWhenUninstallingCorePlugin(@TempDirectory.TempDir Path temp) {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
when(mailPlugin.getDirectory()).thenReturn(temp);
|
||||||
|
when(mailPlugin.isCore()).thenReturn(true);
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(singletonList(mailPlugin));
|
||||||
|
|
||||||
|
assertThrows(ScmConstraintViolationException.class, () -> manager.uninstall("scm-mail-plugin", false));
|
||||||
|
|
||||||
|
assertThat(temp.resolve("uninstall")).doesNotExist();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldMarkUninstallablePlugins() {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
InstalledPlugin reviewPlugin = createInstalled("scm-review-plugin");
|
||||||
|
when(reviewPlugin.getDescriptor().getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(asList(mailPlugin, reviewPlugin));
|
||||||
|
|
||||||
|
manager.computeInstallationDependencies();
|
||||||
|
|
||||||
|
verify(reviewPlugin).setUninstallable(true);
|
||||||
|
verify(mailPlugin).setUninstallable(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldUpdateMayUninstallFlagAfterDependencyIsUninstalled() {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
InstalledPlugin reviewPlugin = createInstalled("scm-review-plugin");
|
||||||
|
when(reviewPlugin.getDescriptor().getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(asList(mailPlugin, reviewPlugin));
|
||||||
|
|
||||||
|
manager.computeInstallationDependencies();
|
||||||
|
|
||||||
|
manager.uninstall("scm-review-plugin", false);
|
||||||
|
|
||||||
|
verify(mailPlugin).setUninstallable(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldUpdateMayUninstallFlagAfterDependencyIsInstalled() {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
AvailablePlugin reviewPlugin = createAvailable("scm-review-plugin");
|
||||||
|
when(reviewPlugin.getDescriptor().getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(singletonList(mailPlugin));
|
||||||
|
when(center.getAvailable()).thenReturn(singleton(reviewPlugin));
|
||||||
|
|
||||||
|
manager.computeInstallationDependencies();
|
||||||
|
|
||||||
|
manager.install("scm-review-plugin", false);
|
||||||
|
|
||||||
|
verify(mailPlugin).setUninstallable(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldRestartWithUninstallOnly() {
|
||||||
|
InstalledPlugin mailPlugin = createInstalled("scm-mail-plugin");
|
||||||
|
when(mailPlugin.isMarkedForUninstall()).thenReturn(true);
|
||||||
|
|
||||||
|
when(loader.getInstalledPlugins()).thenReturn(singletonList(mailPlugin));
|
||||||
|
|
||||||
|
manager.executePendingAndRestart();
|
||||||
|
|
||||||
|
verify(eventBus).post(any(RestartEvent.class));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nested
|
@Nested
|
||||||
@@ -349,38 +473,14 @@ class DefaultPluginManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldThrowAuthorizationExceptionsForInstallPendingAndRestart() {
|
void shouldThrowAuthorizationExceptionsForUninstallMethod() {
|
||||||
assertThrows(AuthorizationException.class, () -> manager.installPendingAndRestart());
|
assertThrows(AuthorizationException.class, () -> manager.uninstall("test", false));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldThrowAuthorizationExceptionsForExecutePendingAndRestart() {
|
||||||
|
assertThrows(AuthorizationException.class, () -> manager.executePendingAndRestart());
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private AvailablePlugin createAvailable(String name) {
|
|
||||||
PluginInformation information = new PluginInformation();
|
|
||||||
information.setName(name);
|
|
||||||
return createAvailable(information);
|
|
||||||
}
|
|
||||||
|
|
||||||
private InstalledPlugin createInstalled(String name) {
|
|
||||||
PluginInformation information = new PluginInformation();
|
|
||||||
information.setName(name);
|
|
||||||
return createInstalled(information);
|
|
||||||
}
|
|
||||||
|
|
||||||
private InstalledPlugin createInstalled(PluginInformation information) {
|
|
||||||
InstalledPlugin plugin = mock(InstalledPlugin.class, Answers.RETURNS_DEEP_STUBS);
|
|
||||||
returnInformation(plugin, information);
|
|
||||||
return plugin;
|
|
||||||
}
|
|
||||||
|
|
||||||
private AvailablePlugin createAvailable(PluginInformation information) {
|
|
||||||
AvailablePluginDescriptor descriptor = mock(AvailablePluginDescriptor.class);
|
|
||||||
lenient().when(descriptor.getInformation()).thenReturn(information);
|
|
||||||
return new AvailablePlugin(descriptor);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void returnInformation(Plugin mockedPlugin, PluginInformation information) {
|
|
||||||
when(mockedPlugin.getDescriptor().getInformation()).thenReturn(information);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -248,7 +248,7 @@ public class DefaultUberWebResourceLoaderTest extends WebResourceLoaderTestBase
|
|||||||
private InstalledPlugin createPluginWrapper(Path directory)
|
private InstalledPlugin createPluginWrapper(Path directory)
|
||||||
{
|
{
|
||||||
return new InstalledPlugin(null, null, new PathWebResourceLoader(directory),
|
return new InstalledPlugin(null, null, new PathWebResourceLoader(directory),
|
||||||
directory);
|
directory, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
//~--- fields ---------------------------------------------------------------
|
//~--- fields ---------------------------------------------------------------
|
||||||
|
|||||||
@@ -0,0 +1,81 @@
|
|||||||
|
package sonia.scm.plugin;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Assertions;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import sonia.scm.ScmConstraintViolationException;
|
||||||
|
|
||||||
|
import static java.util.Collections.emptySet;
|
||||||
|
import static java.util.Collections.singleton;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
import static sonia.scm.plugin.PluginTestHelper.createInstalled;
|
||||||
|
|
||||||
|
class PluginDependencyTrackerTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void simpleInstalledPluginWithoutDependingPluginsCanBeUninstalled() {
|
||||||
|
PluginDescriptor mail = createInstalled("scm-mail-plugin").getDescriptor();
|
||||||
|
when(mail.getDependencies()).thenReturn(emptySet());
|
||||||
|
|
||||||
|
PluginDependencyTracker pluginDependencyTracker = new PluginDependencyTracker();
|
||||||
|
pluginDependencyTracker.addInstalled(mail);
|
||||||
|
|
||||||
|
boolean mayUninstall = pluginDependencyTracker.mayUninstall("scm-mail-plugin");
|
||||||
|
assertThat(mayUninstall).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void installedPluginWithDependingPluginCannotBeUninstalled() {
|
||||||
|
PluginDescriptor review = createInstalled("scm-review-plugin").getDescriptor();
|
||||||
|
when(review.getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
PluginDependencyTracker pluginDependencyTracker = new PluginDependencyTracker();
|
||||||
|
pluginDependencyTracker.addInstalled(review);
|
||||||
|
|
||||||
|
boolean mayUninstall = pluginDependencyTracker.mayUninstall("scm-mail-plugin");
|
||||||
|
assertThat(mayUninstall).isFalse();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void uninstallOfRequiredPluginShouldThrowException() {
|
||||||
|
PluginDescriptor review = createInstalled("scm-review-plugin").getDescriptor();
|
||||||
|
when(review.getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
PluginDependencyTracker pluginDependencyTracker = new PluginDependencyTracker();
|
||||||
|
pluginDependencyTracker.addInstalled(review);
|
||||||
|
|
||||||
|
Assertions.assertThrows(
|
||||||
|
ScmConstraintViolationException.class,
|
||||||
|
() -> pluginDependencyTracker.removeInstalled(createInstalled("scm-mail-plugin").getDescriptor())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void installedPluginWithDependingPluginCanBeUninstalledAfterDependingPluginIsUninstalled() {
|
||||||
|
PluginDescriptor review = createInstalled("scm-review-plugin").getDescriptor();
|
||||||
|
when(review.getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
PluginDependencyTracker pluginDependencyTracker = new PluginDependencyTracker();
|
||||||
|
pluginDependencyTracker.addInstalled(review);
|
||||||
|
pluginDependencyTracker.removeInstalled(review);
|
||||||
|
|
||||||
|
boolean mayUninstall = pluginDependencyTracker.mayUninstall("scm-mail-plugin");
|
||||||
|
assertThat(mayUninstall).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void installedPluginWithMultipleDependingPluginCannotBeUninstalledAfterOnlyOneDependingPluginIsUninstalled() {
|
||||||
|
PluginDescriptor review = createInstalled("scm-review-plugin").getDescriptor();
|
||||||
|
when(review.getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
PluginDescriptor jira = createInstalled("scm-jira-plugin").getDescriptor();
|
||||||
|
when(jira.getDependencies()).thenReturn(singleton("scm-mail-plugin"));
|
||||||
|
|
||||||
|
PluginDependencyTracker pluginDependencyTracker = new PluginDependencyTracker();
|
||||||
|
pluginDependencyTracker.addInstalled(review);
|
||||||
|
pluginDependencyTracker.addInstalled(jira);
|
||||||
|
pluginDependencyTracker.removeInstalled(review);
|
||||||
|
|
||||||
|
boolean mayUninstall = pluginDependencyTracker.mayUninstall("scm-mail-plugin");
|
||||||
|
assertThat(mayUninstall).isFalse();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
package sonia.scm.plugin;
|
||||||
|
|
||||||
|
import org.mockito.Answers;
|
||||||
|
|
||||||
|
import static org.mockito.Mockito.lenient;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
public class PluginTestHelper {
|
||||||
|
public static AvailablePlugin createAvailable(String name) {
|
||||||
|
PluginInformation information = new PluginInformation();
|
||||||
|
information.setName(name);
|
||||||
|
information.setVersion("1.0");
|
||||||
|
return createAvailable(information);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static InstalledPlugin createInstalled(String name) {
|
||||||
|
PluginInformation information = new PluginInformation();
|
||||||
|
information.setName(name);
|
||||||
|
information.setVersion("1.0");
|
||||||
|
return createInstalled(information);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static InstalledPlugin createInstalled(PluginInformation information) {
|
||||||
|
InstalledPlugin plugin = mock(InstalledPlugin.class, Answers.RETURNS_DEEP_STUBS);
|
||||||
|
returnInformation(plugin, information);
|
||||||
|
return plugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AvailablePlugin createAvailable(PluginInformation information) {
|
||||||
|
AvailablePluginDescriptor descriptor = mock(AvailablePluginDescriptor.class);
|
||||||
|
lenient().when(descriptor.getInformation()).thenReturn(information);
|
||||||
|
return new AvailablePlugin(descriptor);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void returnInformation(Plugin mockedPlugin, PluginInformation information) {
|
||||||
|
when(mockedPlugin.getDescriptor().getInformation()).thenReturn(information);
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user