| | |
| | | */
|
| | | package com.gitblit.wicket.panels;
|
| | |
|
| | | import java.io.Serializable;
|
| | | import java.text.MessageFormat;
|
| | | import java.util.ArrayList;
|
| | | import java.util.HashMap;
|
| | | import java.util.List;
|
| | | import java.util.Map;
|
| | |
|
| | | import javax.servlet.http.HttpServletRequest;
|
| | |
|
| | | import org.apache.wicket.Component;
|
| | | import org.apache.wicket.Localizer;
|
| | | import org.apache.wicket.RequestCycle;
|
| | | import org.apache.wicket.markup.html.basic.Label;
|
| | | import org.apache.wicket.markup.html.image.ContextImage;
|
| | | import org.apache.wicket.markup.html.panel.Fragment;
|
| | | import org.apache.wicket.markup.repeater.Item;
|
| | | import org.apache.wicket.markup.repeater.data.DataView;
|
| | | import org.apache.wicket.markup.repeater.data.ListDataProvider;
|
| | | import org.apache.wicket.protocol.http.WebRequest;
|
| | | import org.apache.wicket.protocol.http.request.WebClientInfo;
|
| | |
|
| | | import com.gitblit.Constants;
|
| | | import com.gitblit.Constants.AccessPermission;
|
| | | import com.gitblit.Constants.AccessRestrictionType;
|
| | | import com.gitblit.GitBlit;
|
| | | import com.gitblit.Keys;
|
| | | import com.gitblit.SparkleShareInviteServlet;
|
| | | import com.gitblit.models.GitClientApplication;
|
| | | import com.gitblit.models.RepositoryModel;
|
| | | import com.gitblit.models.RepositoryUrl;
|
| | | import com.gitblit.models.UserModel;
|
| | | import com.gitblit.utils.StringUtils;
|
| | | import com.gitblit.wicket.ExternalImage;
|
| | | import com.gitblit.wicket.GitBlitWebSession;
|
| | | import com.gitblit.wicket.WicketUtils;
|
| | |
|
| | | /**
|
| | | * Smart repository url panel which can display multiple Gitblit repository urls
|
| | | * and also supports 3rd party app clone links.
|
| | | * |
| | | *
|
| | | * @author James Moger
|
| | | *
|
| | | */
|
| | | public class RepositoryUrlPanel extends BasePanel {
|
| | |
|
| | | private static final long serialVersionUID = 1L;
|
| | | |
| | | private final String primaryUrl;
|
| | |
|
| | | public RepositoryUrlPanel(String wicketId, boolean onlyPrimary, UserModel user, |
| | | RepositoryModel repository, Localizer localizer, Component owner) {
|
| | | private final String externalPermission = "?";
|
| | |
|
| | | private boolean onlyUrls;
|
| | | private UserModel user;
|
| | | private RepositoryModel repository;
|
| | | private RepositoryUrl primaryUrl;
|
| | | private Map<String, String> urlPermissionsMap;
|
| | | private Map<AccessRestrictionType, String> accessRestrictionsMap;
|
| | |
|
| | | public RepositoryUrlPanel(String wicketId, boolean onlyUrls, UserModel user, RepositoryModel repository) {
|
| | | super(wicketId);
|
| | | if (user == null) {
|
| | | user = UserModel.ANONYMOUS;
|
| | | }
|
| | | List<String> repositoryUrls = new ArrayList<String>();
|
| | | this.onlyUrls = onlyUrls;
|
| | | this.user = user == null ? UserModel.ANONYMOUS : user;
|
| | | this.repository = repository;
|
| | | this.urlPermissionsMap = new HashMap<String, String>();
|
| | | }
|
| | |
|
| | | AccessPermission accessPermission = null;
|
| | | if (GitBlit.getBoolean(Keys.git.enableGitServlet, true)) {
|
| | | accessPermission = user.getRepositoryPermission(repository).permission;
|
| | | repositoryUrls.add(getRepositoryUrl(repository));
|
| | | }
|
| | | repositoryUrls.addAll(GitBlit.self().getOtherCloneUrls(repository.name, UserModel.ANONYMOUS.equals(user) ? "" : user.username));
|
| | | |
| | | primaryUrl = repositoryUrls.size() == 0 ? "" : repositoryUrls.remove(0);
|
| | | @Override
|
| | | protected void onInitialize() {
|
| | | super.onInitialize();
|
| | |
|
| | | add(new DetailedRepositoryUrlPanel("repositoryPrimaryUrl", localizer, owner, repository.name, primaryUrl, accessPermission));
|
| | | |
| | | if (!onlyPrimary) {
|
| | | Component gitDaemonUrlPanel = createGitDaemonUrlPanel("repositoryGitDaemonUrl", user, repository);
|
| | | if (!StringUtils.isEmpty(primaryUrl) && gitDaemonUrlPanel instanceof DetailedRepositoryUrlPanel) {
|
| | | WicketUtils.setCssStyle(gitDaemonUrlPanel, "padding-top: 10px");
|
| | | }
|
| | | add(gitDaemonUrlPanel);
|
| | | HttpServletRequest req = ((WebRequest) getRequest()).getHttpServletRequest();
|
| | |
|
| | | List<RepositoryUrl> repositoryUrls = app().services().getRepositoryUrls(req, user, repository);
|
| | | // grab primary url from the top of the list
|
| | | primaryUrl = repositoryUrls.size() == 0 ? null : repositoryUrls.get(0);
|
| | |
|
| | | boolean canClone = primaryUrl != null && (!primaryUrl.hasPermission() || primaryUrl.permission.atLeast(AccessPermission.CLONE));
|
| | |
|
| | | if (repositoryUrls.size() == 0 || !canClone) {
|
| | | // no urls, nothing to show.
|
| | | add(new Label("repositoryUrlPanel").setVisible(false));
|
| | | add(new Label("applicationMenusPanel").setVisible(false));
|
| | | add(new Label("repositoryIndicators").setVisible(false));
|
| | | return;
|
| | | }
|
| | |
|
| | | // display primary url
|
| | | add(createPrimaryUrlPanel("repositoryUrlPanel", repository, repositoryUrls));
|
| | |
|
| | | if (onlyUrls) {
|
| | | add(new Label("repositoryIndicators").setVisible(false));
|
| | | } else {
|
| | | add(new Label("repositoryGitDaemonUrl").setVisible(false));
|
| | | }
|
| | | |
| | | String cloneWith = localizer.getString("gb.cloneWithApp", owner);
|
| | | final List<AppCloneLink> cloneLinks = new ArrayList<AppCloneLink>();
|
| | | if (user.canClone(repository) && GitBlit.getBoolean(Keys.web.allowAppCloneLinks, true)) {
|
| | | // universal app clone urls
|
| | | // cloneLinks.add(new AppCloneLink(MessageFormat.format(cloneWith, "SmartGit\u2122"),
|
| | | // MessageFormat.format("smartgit://cloneRepo/{0}", primaryUrl),
|
| | | // "Syntevo SmartGit\u2122"));
|
| | |
|
| | | if (isWindows()) {
|
| | | // Windows client app clone urls
|
| | | cloneLinks.add(new AppCloneLink(MessageFormat.format(cloneWith, "SourceTree\u2122"),
|
| | | MessageFormat.format("sourcetree://cloneRepo/{0}", primaryUrl),
|
| | | "Atlassian SourceTree\u2122"));
|
| | | // cloneLinks.add(new AppCloneLink(
|
| | | // MessageFormat.format(cloneWith, "GitHub\u2122 for Windows"),
|
| | | // MessageFormat.format("github-windows://openRepo/{0}", primaryUrl),
|
| | | // "GitHub\u2122 for Windows"));
|
| | | } else if (isMac()) {
|
| | | // Mac client app clone urls
|
| | | cloneLinks.add(new AppCloneLink(MessageFormat.format(cloneWith, "SourceTree\u2122"),
|
| | | MessageFormat.format("sourcetree://cloneRepo/{0}", primaryUrl),
|
| | | "Atlassian SourceTree\u2122"));
|
| | | // cloneLinks.add(new AppCloneLink(
|
| | | // MessageFormat.format(cloneWith, "GitHub\u2122 for Mac"),
|
| | | // MessageFormat.format("github-mac://openRepo/{0}", primaryUrl),
|
| | | // "GitHub\u2122 for Mac"));
|
| | | }
|
| | |
|
| | | // sparkleshare invite url
|
| | | String sparkleshareUrl = getSparkleShareInviteUrl(user, repository);
|
| | | if (!StringUtils.isEmpty(sparkleshareUrl)) {
|
| | | cloneLinks.add(new AppCloneLink(MessageFormat.format(cloneWith, "SparkleShare\u2122"),
|
| | | sparkleshareUrl, "SparkleShare\u2122", "icon-star"));
|
| | | }
|
| | | add(createRepositoryIndicators(repository));
|
| | | }
|
| | |
|
| | | // app clone links
|
| | | ListDataProvider<AppCloneLink> appLinks = new ListDataProvider<AppCloneLink>(cloneLinks);
|
| | | DataView<AppCloneLink> appCloneLinks = new DataView<AppCloneLink>("appCloneLink", appLinks) {
|
| | | private static final long serialVersionUID = 1L;
|
| | | int count;
|
| | | |
| | | public void populateItem(final Item<AppCloneLink> item) {
|
| | | final AppCloneLink appLink = item.getModelObject();
|
| | | item.add(new Label("icon", MessageFormat.format("<i class=\"{0}\"></i>", appLink.icon)).setEscapeModelStrings(false));
|
| | | LinkPanel linkPanel = new LinkPanel("link", null, appLink.name, appLink.url);
|
| | | if (!StringUtils.isEmpty(appLink.tooltip)) {
|
| | | WicketUtils.setHtmlTooltip(linkPanel, appLink.tooltip);
|
| | | boolean allowAppLinks = app().settings().getBoolean(Keys.web.allowAppCloneLinks, true);
|
| | | if (onlyUrls || !canClone || !allowAppLinks) {
|
| | | // only display the url(s)
|
| | | add(new Label("applicationMenusPanel").setVisible(false));
|
| | | return;
|
| | | }
|
| | | // create the git client application menus
|
| | | add(createApplicationMenus("applicationMenusPanel", user, repository, repositoryUrls));
|
| | | }
|
| | |
|
| | | public String getPrimaryUrl() {
|
| | | return primaryUrl == null ? "" : primaryUrl.url;
|
| | | }
|
| | |
|
| | | protected Fragment createPrimaryUrlPanel(String wicketId, final RepositoryModel repository, List<RepositoryUrl> repositoryUrls) {
|
| | |
|
| | | Fragment urlPanel = new Fragment(wicketId, "repositoryUrlFragment", this);
|
| | | urlPanel.setRenderBodyOnly(true);
|
| | |
|
| | | if (repositoryUrls.size() == 1) {
|
| | | //
|
| | | // Single repository url, no dropdown menu
|
| | | //
|
| | | urlPanel.add(new Label("menu").setVisible(false));
|
| | | } else {
|
| | | //
|
| | | // Multiple repository urls, show url drop down menu
|
| | | //
|
| | | ListDataProvider<RepositoryUrl> urlsDp = new ListDataProvider<RepositoryUrl>(repositoryUrls);
|
| | | DataView<RepositoryUrl> repoUrlMenuItems = new DataView<RepositoryUrl>("repoUrls", urlsDp) {
|
| | | private static final long serialVersionUID = 1L;
|
| | |
|
| | | @Override
|
| | | public void populateItem(final Item<RepositoryUrl> item) {
|
| | | RepositoryUrl repoUrl = item.getModelObject();
|
| | | // repository url
|
| | | Fragment fragment = new Fragment("repoUrl", "actionFragment", this);
|
| | | Component content = new Label("content", repoUrl.url).setRenderBodyOnly(true);
|
| | | WicketUtils.setCssClass(content, "commandMenuItem");
|
| | | fragment.add(content);
|
| | | item.add(fragment);
|
| | |
|
| | | Label permissionLabel = new Label("permission", repoUrl.hasPermission() ? repoUrl.permission.toString() : externalPermission);
|
| | | WicketUtils.setPermissionClass(permissionLabel, repoUrl.permission);
|
| | | String tooltip = getProtocolPermissionDescription(repository, repoUrl);
|
| | | WicketUtils.setHtmlTooltip(permissionLabel, tooltip);
|
| | | fragment.add(permissionLabel);
|
| | | fragment.add(createCopyFragment(repoUrl.url));
|
| | | }
|
| | | item.add(linkPanel);
|
| | | item.add(new Label("separator", "|").setVisible(count < (cloneLinks.size() - 1)));
|
| | | count++;
|
| | | };
|
| | |
|
| | | Fragment urlMenuFragment = new Fragment("menu", "urlProtocolMenuFragment", this);
|
| | | urlMenuFragment.setRenderBodyOnly(true);
|
| | | urlMenuFragment.add(new Label("menuText", getString("gb.url")));
|
| | | urlMenuFragment.add(repoUrlMenuItems);
|
| | | urlPanel.add(urlMenuFragment);
|
| | | }
|
| | |
|
| | | // access restriction icon and tooltip
|
| | | if (repository.isMirror) {
|
| | | urlPanel.add(WicketUtils.newImage("accessRestrictionIcon", "mirror_16x16.png",
|
| | | getString("gb.isMirror")));
|
| | | } else if (app().services().isServingRepositories()) {
|
| | | switch (repository.accessRestriction) {
|
| | | case NONE:
|
| | | urlPanel.add(WicketUtils.newClearPixel("accessRestrictionIcon").setVisible(false));
|
| | | break;
|
| | | case PUSH:
|
| | | urlPanel.add(WicketUtils.newImage("accessRestrictionIcon", "lock_go_16x16.png",
|
| | | getAccessRestrictions().get(repository.accessRestriction)));
|
| | | break;
|
| | | case CLONE:
|
| | | urlPanel.add(WicketUtils.newImage("accessRestrictionIcon", "lock_pull_16x16.png",
|
| | | getAccessRestrictions().get(repository.accessRestriction)));
|
| | | break;
|
| | | case VIEW:
|
| | | urlPanel.add(WicketUtils.newImage("accessRestrictionIcon", "shield_16x16.png",
|
| | | getAccessRestrictions().get(repository.accessRestriction)));
|
| | | break;
|
| | | default:
|
| | | if (repositoryUrls.size() == 1) {
|
| | | // force left end cap to have some width
|
| | | urlPanel.add(WicketUtils.newBlankIcon("accessRestrictionIcon"));
|
| | | } else {
|
| | | urlPanel.add(WicketUtils.newClearPixel("accessRestrictionIcon").setVisible(false));
|
| | | }
|
| | | }
|
| | | } else {
|
| | | if (repositoryUrls.size() == 1) {
|
| | | // force left end cap to have some width
|
| | | urlPanel.add(WicketUtils.newBlankIcon("accessRestrictionIcon"));
|
| | | } else {
|
| | | urlPanel.add(WicketUtils.newClearPixel("accessRestrictionIcon").setVisible(false));
|
| | | }
|
| | | }
|
| | |
|
| | | urlPanel.add(new Label("primaryUrl", primaryUrl.url).setRenderBodyOnly(true));
|
| | |
|
| | | Label permissionLabel = new Label("primaryUrlPermission", primaryUrl.hasPermission() ? primaryUrl.permission.toString() : externalPermission);
|
| | | String tooltip = getProtocolPermissionDescription(repository, primaryUrl);
|
| | | WicketUtils.setHtmlTooltip(permissionLabel, tooltip);
|
| | | urlPanel.add(permissionLabel);
|
| | | urlPanel.add(createCopyFragment(primaryUrl.url));
|
| | |
|
| | | return urlPanel;
|
| | | }
|
| | |
|
| | | protected Fragment createApplicationMenus(String wicketId, final UserModel user, final RepositoryModel repository, final List<RepositoryUrl> repositoryUrls) {
|
| | | final List<GitClientApplication> displayedApps = new ArrayList<GitClientApplication>();
|
| | | final String userAgent = ((WebClientInfo) GitBlitWebSession.get().getClientInfo()).getUserAgent();
|
| | |
|
| | | if (user.canClone(repository)) {
|
| | | for (GitClientApplication app : app().gitblit().getClientApplications()) {
|
| | | if (app.isActive && app.allowsPlatform(userAgent)) {
|
| | | displayedApps.add(app);
|
| | | }
|
| | | }
|
| | | }
|
| | |
|
| | | final String baseURL = WicketUtils.getGitblitURL(RequestCycle.get().getRequest());
|
| | | ListDataProvider<GitClientApplication> displayedAppsDp = new ListDataProvider<GitClientApplication>(displayedApps);
|
| | | DataView<GitClientApplication> appMenus = new DataView<GitClientApplication>("appMenus", displayedAppsDp) {
|
| | | private static final long serialVersionUID = 1L;
|
| | |
|
| | | @Override
|
| | | public void populateItem(final Item<GitClientApplication> item) {
|
| | | final GitClientApplication clientApp = item.getModelObject();
|
| | |
|
| | | // filter the urls for the client app
|
| | | List<RepositoryUrl> urls = new ArrayList<RepositoryUrl>();
|
| | | for (RepositoryUrl repoUrl : repositoryUrls) {
|
| | | if (clientApp.minimumPermission == null || !repoUrl.hasPermission()) {
|
| | | // no minimum permission or untracked permissions, assume it is satisfactory
|
| | | if (clientApp.supportsTransport(repoUrl.url)) {
|
| | | urls.add(repoUrl);
|
| | | }
|
| | | } else if (repoUrl.permission.atLeast(clientApp.minimumPermission)) {
|
| | | // repo url meets minimum permission requirement
|
| | | if (clientApp.supportsTransport(repoUrl.url)) {
|
| | | urls.add(repoUrl);
|
| | | }
|
| | | }
|
| | | }
|
| | |
|
| | | if (urls.size() == 0) {
|
| | | // do not show this app menu because there are no urls
|
| | | item.add(new Label("appMenu").setVisible(false));
|
| | | return;
|
| | | }
|
| | |
|
| | | Fragment appMenu = new Fragment("appMenu", "appMenuFragment", this);
|
| | | appMenu.setRenderBodyOnly(true);
|
| | | item.add(appMenu);
|
| | |
|
| | | // menu button
|
| | | appMenu.add(new Label("applicationName", clientApp.name));
|
| | |
|
| | | // application icon
|
| | | Component img;
|
| | | if (StringUtils.isEmpty(clientApp.icon)) {
|
| | | img = WicketUtils.newClearPixel("applicationIcon").setVisible(false);
|
| | | } else {
|
| | | if (clientApp.icon.contains("://")) {
|
| | | // external image
|
| | | img = new ExternalImage("applicationIcon", clientApp.icon);
|
| | | } else {
|
| | | // context image
|
| | | img = WicketUtils.newImage("applicationIcon", clientApp.icon);
|
| | | }
|
| | | }
|
| | | appMenu.add(img);
|
| | |
|
| | | // application menu title, may be a link
|
| | | if (StringUtils.isEmpty(clientApp.productUrl)) {
|
| | | appMenu.add(new Label("applicationTitle", clientApp.toString()));
|
| | | } else {
|
| | | appMenu.add(new LinkPanel("applicationTitle", null, clientApp.toString(), clientApp.productUrl, true));
|
| | | }
|
| | |
|
| | | // brief application description
|
| | | if (StringUtils.isEmpty(clientApp.description)) {
|
| | | appMenu.add(new Label("applicationDescription").setVisible(false));
|
| | | } else {
|
| | | appMenu.add(new Label("applicationDescription", clientApp.description));
|
| | | }
|
| | |
|
| | | // brief application legal info, copyright, license, etc
|
| | | if (StringUtils.isEmpty(clientApp.legal)) {
|
| | | appMenu.add(new Label("applicationLegal").setVisible(false));
|
| | | } else {
|
| | | appMenu.add(new Label("applicationLegal", clientApp.legal));
|
| | | }
|
| | |
|
| | | // a nested repeater for all action items
|
| | | ListDataProvider<RepositoryUrl> urlsDp = new ListDataProvider<RepositoryUrl>(urls);
|
| | | DataView<RepositoryUrl> actionItems = new DataView<RepositoryUrl>("actionItems", urlsDp) {
|
| | | private static final long serialVersionUID = 1L;
|
| | |
|
| | | @Override
|
| | | public void populateItem(final Item<RepositoryUrl> repoLinkItem) {
|
| | | RepositoryUrl repoUrl = repoLinkItem.getModelObject();
|
| | | Fragment fragment = new Fragment("actionItem", "actionFragment", this);
|
| | | fragment.add(createPermissionBadge("permission", repoUrl));
|
| | |
|
| | | if (!StringUtils.isEmpty(clientApp.cloneUrl)) {
|
| | | // custom registered url
|
| | | String url = substitute(clientApp.cloneUrl, repoUrl.url, baseURL, user.username, repository.name);
|
| | | fragment.add(new LinkPanel("content", "applicationMenuItem", getString("gb.clone") + " " + repoUrl.url, url));
|
| | | repoLinkItem.add(fragment);
|
| | | fragment.add(new Label("copyFunction").setVisible(false));
|
| | | } else if (!StringUtils.isEmpty(clientApp.command)) {
|
| | | // command-line
|
| | | String command = substitute(clientApp.command, repoUrl.url, baseURL, user.username, repository.name);
|
| | | Label content = new Label("content", command);
|
| | | WicketUtils.setCssClass(content, "commandMenuItem");
|
| | | fragment.add(content);
|
| | | repoLinkItem.add(fragment);
|
| | |
|
| | | // copy function for command
|
| | | fragment.add(createCopyFragment(command));
|
| | | }
|
| | | }};
|
| | | appMenu.add(actionItems);
|
| | | }
|
| | | };
|
| | | add(appCloneLinks);
|
| | |
|
| | | Fragment applicationMenus = new Fragment(wicketId, "applicationMenusFragment", this);
|
| | | applicationMenus.add(appMenus);
|
| | | return applicationMenus;
|
| | | }
|
| | | |
| | | public String getPrimaryUrl() {
|
| | | return primaryUrl;
|
| | |
|
| | | protected String substitute(String pattern, String repoUrl, String baseUrl, String username, String repository) {
|
| | | return pattern.replace("${repoUrl}", repoUrl).replace("${baseUrl}", baseUrl).replace("${username}", username).replace("${repository}", repository);
|
| | | }
|
| | | |
| | | protected String getRepositoryUrl(RepositoryModel repository) {
|
| | | StringBuilder sb = new StringBuilder();
|
| | | sb.append(WicketUtils.getGitblitURL(RequestCycle.get().getRequest()));
|
| | | sb.append(Constants.GIT_PATH);
|
| | | sb.append(repository.name);
|
| | | |
| | | // inject username into repository url if authentication is required
|
| | | if (repository.accessRestriction.exceeds(AccessRestrictionType.NONE)
|
| | | && GitBlitWebSession.get().isLoggedIn()) {
|
| | | String username = GitBlitWebSession.get().getUsername();
|
| | | sb.insert(sb.indexOf("://") + 3, username + "@");
|
| | | }
|
| | | return sb.toString();
|
| | |
|
| | | protected Label createPermissionBadge(String wicketId, RepositoryUrl repoUrl) {
|
| | | Label permissionLabel = new Label(wicketId, repoUrl.hasPermission() ? repoUrl.permission.toString() : externalPermission);
|
| | | WicketUtils.setPermissionClass(permissionLabel, repoUrl.permission);
|
| | | String tooltip = getProtocolPermissionDescription(repository, repoUrl);
|
| | | WicketUtils.setHtmlTooltip(permissionLabel, tooltip);
|
| | | return permissionLabel;
|
| | | }
|
| | | |
| | | protected Component createGitDaemonUrlPanel(String wicketId, UserModel user, RepositoryModel repository) {
|
| | | int gitDaemonPort = GitBlit.getInteger(Keys.git.daemonPort, 0);
|
| | | if (gitDaemonPort > 0 && user.canClone(repository)) {
|
| | | String servername = ((WebRequest) getRequest()).getHttpServletRequest().getServerName();
|
| | | String gitDaemonUrl;
|
| | | if (gitDaemonPort == 9418) {
|
| | | // standard port
|
| | | gitDaemonUrl = MessageFormat.format("git://{0}/{1}", servername, repository.name);
|
| | | } else {
|
| | | // non-standard port
|
| | | gitDaemonUrl = MessageFormat.format("git://{0}:{1,number,0}/{2}", servername, gitDaemonPort, repository.name);
|
| | | }
|
| | | |
| | | AccessPermission gitDaemonPermission = user.getRepositoryPermission(repository).permission;;
|
| | | if (gitDaemonPermission.atLeast(AccessPermission.CLONE)) {
|
| | | if (repository.accessRestriction.atLeast(AccessRestrictionType.CLONE)) {
|
| | | // can not authenticate clone via anonymous git protocol
|
| | | gitDaemonPermission = AccessPermission.NONE;
|
| | | } else if (repository.accessRestriction.atLeast(AccessRestrictionType.PUSH)) {
|
| | | // can not authenticate push via anonymous git protocol
|
| | | gitDaemonPermission = AccessPermission.CLONE;
|
| | | } else {
|
| | | // normal user permission
|
| | | }
|
| | | }
|
| | | |
| | | if (AccessPermission.NONE.equals(gitDaemonPermission)) {
|
| | | // repository prohibits all anonymous access
|
| | | return new Label(wicketId).setVisible(false);
|
| | | } else {
|
| | | // repository allows some form of anonymous access
|
| | | return new DetailedRepositoryUrlPanel(wicketId, getLocalizer(), this, repository.name, gitDaemonUrl, gitDaemonPermission);
|
| | | }
|
| | |
|
| | | protected Fragment createCopyFragment(String text) {
|
| | | if (app().settings().getBoolean(Keys.web.allowFlashCopyToClipboard, true)) {
|
| | | // clippy: flash-based copy & paste
|
| | | Fragment copyFragment = new Fragment("copyFunction", "clippyPanel", this);
|
| | | String baseUrl = WicketUtils.getGitblitURL(getRequest());
|
| | | ShockWaveComponent clippy = new ShockWaveComponent("clippy", baseUrl + "/clippy.swf");
|
| | | clippy.setValue("flashVars", "text=" + StringUtils.encodeURL(text));
|
| | | copyFragment.add(clippy);
|
| | | return copyFragment;
|
| | | } else {
|
| | | // git daemon is not running
|
| | | return new Label(wicketId).setVisible(false);
|
| | | // javascript: manual copy & paste with modal browser prompt dialog
|
| | | Fragment copyFragment = new Fragment("copyFunction", "jsPanel", this);
|
| | | ContextImage img = WicketUtils.newImage("copyIcon", "clippy.png");
|
| | | img.add(new JavascriptTextPrompt("onclick", "Copy to Clipboard (Ctrl+C, Enter)", text));
|
| | | copyFragment.add(img);
|
| | | return copyFragment;
|
| | | }
|
| | | }
|
| | |
|
| | | protected String getSparkleShareInviteUrl(UserModel user, RepositoryModel repository) {
|
| | | if (repository.isBare && repository.isSparkleshared()) {
|
| | | String username = null;
|
| | | if (UserModel.ANONYMOUS != user) {
|
| | | username = user.username;
|
| | | }
|
| | | if (GitBlit.getBoolean(Keys.git.enableGitServlet, true) || (GitBlit.getInteger(Keys.git.daemonPort, 0) > 0)) {
|
| | | // Gitblit as server
|
| | | // ensure user can rewind
|
| | | if (user.canRewindRef(repository)) {
|
| | | String baseURL = WicketUtils.getGitblitURL(RequestCycle.get().getRequest());
|
| | | return SparkleShareInviteServlet.asLink(baseURL, repository.name, username);
|
| | | protected String getProtocolPermissionDescription(RepositoryModel repository,
|
| | | RepositoryUrl repoUrl) {
|
| | | if (!urlPermissionsMap.containsKey(repoUrl.url)) {
|
| | | String note;
|
| | | if (repoUrl.hasPermission()) {
|
| | | note = null;
|
| | | String key;
|
| | | switch (repoUrl.permission) {
|
| | | case OWNER:
|
| | | case REWIND:
|
| | | key = "gb.rewindPermission";
|
| | | break;
|
| | | case DELETE:
|
| | | key = "gb.deletePermission";
|
| | | break;
|
| | | case CREATE:
|
| | | key = "gb.createPermission";
|
| | | break;
|
| | | case PUSH:
|
| | | key = "gb.pushPermission";
|
| | | break;
|
| | | case CLONE:
|
| | | key = "gb.clonePermission";
|
| | | break;
|
| | | default:
|
| | | key = null;
|
| | | note = getString("gb.viewAccess");
|
| | | break;
|
| | | }
|
| | |
|
| | | if (note == null) {
|
| | | String pattern = getString(key);
|
| | | String description = MessageFormat.format(pattern, repoUrl.permission.toString());
|
| | | note = description;
|
| | | }
|
| | | } else {
|
| | | // Gitblit as viewer, assume RW+ permission
|
| | | String baseURL = WicketUtils.getGitblitURL(RequestCycle.get().getRequest());
|
| | | return SparkleShareInviteServlet.asLink(baseURL, repository.name, username);
|
| | | String protocol;
|
| | | int protocolIndex = repoUrl.url.indexOf("://");
|
| | | if (protocolIndex > -1) {
|
| | | // explicit protocol specified
|
| | | protocol = repoUrl.url.substring(0, protocolIndex);
|
| | | } else {
|
| | | // implicit SSH url
|
| | | protocol = "ssh";
|
| | | }
|
| | | note = MessageFormat.format(getString("gb.externalPermissions"), protocol);
|
| | | }
|
| | | urlPermissionsMap.put(repoUrl.url, note);
|
| | | }
|
| | | return urlPermissionsMap.get(repoUrl.url);
|
| | | }
|
| | |
|
| | | protected Map<AccessRestrictionType, String> getAccessRestrictions() {
|
| | | if (accessRestrictionsMap == null) {
|
| | | accessRestrictionsMap = new HashMap<AccessRestrictionType, String>();
|
| | | for (AccessRestrictionType type : AccessRestrictionType.values()) {
|
| | | switch (type) {
|
| | | case NONE:
|
| | | accessRestrictionsMap.put(type, getString("gb.notRestricted"));
|
| | | break;
|
| | | case PUSH:
|
| | | accessRestrictionsMap.put(type, getString("gb.pushRestricted"));
|
| | | break;
|
| | | case CLONE:
|
| | | accessRestrictionsMap.put(type, getString("gb.cloneRestricted"));
|
| | | break;
|
| | | case VIEW:
|
| | | accessRestrictionsMap.put(type, getString("gb.viewRestricted"));
|
| | | break;
|
| | | }
|
| | | }
|
| | | }
|
| | | return null;
|
| | | return accessRestrictionsMap;
|
| | | }
|
| | | |
| | | static class AppCloneLink implements Serializable {
|
| | | |
| | | private static final long serialVersionUID = 1L;
|
| | | |
| | | final String name;
|
| | | final String url;
|
| | | final String tooltip;
|
| | | final String icon;
|
| | | |
| | | public AppCloneLink(String name, String url, String tooltip) {
|
| | | this(name, url, tooltip, "icon-download");
|
| | |
|
| | | protected Component createRepositoryIndicators(RepositoryModel repository) {
|
| | | Fragment fragment = new Fragment("repositoryIndicators", "indicatorsFragment", this);
|
| | | if (repository.isBare) {
|
| | | fragment.add(new Label("workingCopyIndicator").setVisible(false));
|
| | | } else {
|
| | | Fragment wc = new Fragment("workingCopyIndicator", "workingCopyFragment", this);
|
| | | Label lbl = new Label("workingCopy", getString("gb.workingCopy"));
|
| | | WicketUtils.setHtmlTooltip(lbl, getString("gb.workingCopyWarning"));
|
| | | wc.add(lbl);
|
| | | fragment.add(wc);
|
| | | }
|
| | | |
| | | public AppCloneLink(String name, String url, String tooltip, String icon) {
|
| | | this.name = name;
|
| | | this.url = url;
|
| | | this.tooltip = tooltip;
|
| | | this.icon = icon;
|
| | |
|
| | | boolean allowForking = app().settings().getBoolean(Keys.web.allowForking, true);
|
| | | if (!allowForking || user == null || !user.isAuthenticated) {
|
| | | // must be logged-in to fork, hide all fork controls
|
| | | fragment.add(new Label("forksProhibitedIndicator").setVisible(false));
|
| | | } else {
|
| | | String fork = app().repositories().getFork(user.username, repository.name);
|
| | | boolean hasFork = fork != null;
|
| | | boolean canFork = user.canFork(repository);
|
| | |
|
| | | if (hasFork || !canFork) {
|
| | | if (user.canFork() && !repository.allowForks) {
|
| | | // show forks prohibited indicator
|
| | | Fragment wc = new Fragment("forksProhibitedIndicator", "forksProhibitedFragment", this);
|
| | | Label lbl = new Label("forksProhibited", getString("gb.forksProhibited"));
|
| | | WicketUtils.setHtmlTooltip(lbl, getString("gb.forksProhibitedWarning"));
|
| | | wc.add(lbl);
|
| | | fragment.add(wc);
|
| | | } else {
|
| | | // can not fork, no need for forks prohibited indicator
|
| | | fragment.add(new Label("forksProhibitedIndicator").setVisible(false));
|
| | | }
|
| | | } else if (canFork) {
|
| | | // can fork and we do not have one
|
| | | fragment.add(new Label("forksProhibitedIndicator").setVisible(false));
|
| | | }
|
| | | }
|
| | | return fragment;
|
| | | }
|
| | | }
|