Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[REST Auth] API tokens & openhab:users console command #1735

Merged
merged 12 commits into from
Oct 25, 2020
Original file line number Diff line number Diff line change
@@ -0,0 +1,194 @@
/**
* Copyright (c) 2010-2020 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.io.console.internal.extension;

import java.util.List;
import java.util.Optional;
import java.util.Set;

import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.auth.ManagedUser;
import org.openhab.core.auth.User;
import org.openhab.core.auth.UserApiToken;
import org.openhab.core.auth.UserRegistry;
import org.openhab.core.io.console.Console;
import org.openhab.core.io.console.extensions.AbstractConsoleCommandExtension;
import org.openhab.core.io.console.extensions.ConsoleCommandExtension;
import org.osgi.service.component.annotations.Activate;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Reference;

/**
* Console command extension to manage users, sessions and API tokens
*
* @author Yannick Schaus - Initial contribution
*/
@Component(service = ConsoleCommandExtension.class)
@NonNullByDefault
public class UserConsoleCommandExtension extends AbstractConsoleCommandExtension {

private static final String SUBCMD_LIST = "list";
private static final String SUBCMD_ADD = "add";
private static final String SUBCMD_REMOVE = "remove";
private static final String SUBCMD_CHANGEPASSWORD = "changePassword";
private static final String SUBCMD_LISTAPITOKENS = "listApiTokens";
private static final String SUBCMD_ADDAPITOKEN = "addApiToken";
private static final String SUBCMD_RMAPITOKEN = "rmApiToken";
private static final String SUBCMD_CLEARSESSIONS = "clearSessions";

private final UserRegistry userRegistry;

@Activate
public UserConsoleCommandExtension(final @Reference UserRegistry userRegistry) {
super("users", "Access the user registry.");
this.userRegistry = userRegistry;
}

@Override
public List<String> getUsages() {
return List.of(buildCommandUsage(SUBCMD_LIST, "lists all users"),
buildCommandUsage(SUBCMD_ADD + " <userId> <password> <role>",
"adds a new user with the specified role"),
buildCommandUsage(SUBCMD_REMOVE + " <userId>", "removes the given user"),
buildCommandUsage(SUBCMD_CHANGEPASSWORD + " <userId> <newPassword>", "changes the password of a user"),
buildCommandUsage(SUBCMD_LISTAPITOKENS, "lists the API tokens for all users"),
buildCommandUsage(SUBCMD_ADDAPITOKEN + " <userId> <tokenName> <scope>",
"adds a new API token on behalf of the specified user for the specified scope"),
buildCommandUsage(SUBCMD_RMAPITOKEN + " <userId> <tokenName>",
"removes (revokes) the specified API token"),
buildCommandUsage(SUBCMD_CLEARSESSIONS + " <userId>",
"clear the refresh tokens associated with the user (will sign the user out of all sessions)"));
}

@Override
public void execute(String[] args, Console console) {
if (args.length > 0) {
String subCommand = args[0];
switch (subCommand) {
case SUBCMD_LIST:
userRegistry.getAll().forEach(user -> console.println(user.toString()));
break;
case SUBCMD_ADD:
if (args.length == 4) {
User existingUser = userRegistry.get(args[1]);
if (existingUser == null) {
User newUser = userRegistry.register(args[1], args[2], Set.of(args[3]));
console.println(newUser.toString());
console.println("User created.");
} else {
console.println("The user already exists.");
}
} else {
console.printUsage(findUsage(SUBCMD_ADD));
}
break;
case SUBCMD_REMOVE:
if (args.length == 2) {
User user = userRegistry.get(args[1]);
if (user != null) {
userRegistry.remove(user.getName());
console.println("User removed.");
} else {
console.println("User not found.");
}
} else {
console.printUsage(findUsage(SUBCMD_REMOVE));
}
break;
case SUBCMD_CHANGEPASSWORD:
if (args.length == 3) {
User user = userRegistry.get(args[1]);
if (user != null) {
userRegistry.changePassword(user, args[2]);
console.println("Password changed.");
} else {
console.println("User not found.");
}
} else {
console.printUsage(findUsage(SUBCMD_CHANGEPASSWORD));
}
break;
case SUBCMD_LISTAPITOKENS:
userRegistry.getAll().forEach(user -> {
ManagedUser managedUser = (ManagedUser) user;
if (!managedUser.getApiTokens().isEmpty()) {
managedUser.getApiTokens()
.forEach(t -> console.println("user=" + user.toString() + ", " + t.toString()));
}
});
break;
case SUBCMD_ADDAPITOKEN:
if (args.length == 4) {
ManagedUser user = (ManagedUser) userRegistry.get(args[1]);
if (user != null) {
Optional<UserApiToken> userApiToken = user.getApiTokens().stream()
.filter(t -> args[2].equals(t.getName())).findAny();
if (userApiToken.isEmpty()) {
String tokenString = userRegistry.addUserApiToken(user, args[2], args[3]);
console.println(tokenString);
} else {
console.println("Cannot create API token: another one with the same name was found.");
}
} else {
console.println("User not found.");
}
} else {
console.printUsage(findUsage(SUBCMD_ADDAPITOKEN));
}
break;
case SUBCMD_RMAPITOKEN:
if (args.length == 3) {
ManagedUser user = (ManagedUser) userRegistry.get(args[1]);
if (user != null) {
Optional<UserApiToken> userApiToken = user.getApiTokens().stream()
.filter(t -> args[2].equals(t.getName())).findAny();
if (userApiToken.isPresent()) {
userRegistry.removeUserApiToken(user, userApiToken.get());
console.println("API token revoked.");
} else {
console.println("No matching API token found.");
}
} else {
console.println("User not found.");
}
} else {
console.printUsage(findUsage(SUBCMD_RMAPITOKEN));
}
break;
case SUBCMD_CLEARSESSIONS:
if (args.length == 2) {
User user = userRegistry.get(args[1]);
if (user != null) {
userRegistry.clearSessions(user);
console.println("User sessions cleared.");
} else {
console.println("User not found.");
}
} else {
console.printUsage(findUsage(SUBCMD_CLEARSESSIONS));
}
break;
default:
console.println("Unknown command '" + subCommand + "'");
printUsage(console);
break;
}
} else {
printUsage(console);
}
}

private String findUsage(String cmd) {
return getUsages().stream().filter(u -> u.contains(cmd)).findAny().get();
}
}
46 changes: 41 additions & 5 deletions bundles/org.openhab.core.io.http.auth/pages/authorize.html
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,14 @@
margin-bottom: 2rem;
}

form.hide {
display: none;
}

form.show {
display: initial;
}

input {
border: 0;
padding: 10px;
Expand All @@ -54,41 +62,69 @@
margin-bottom: 1rem;
}

input.submit {
.submit {
cursor: pointer;
margin-top: 2rem;
background-color: rgb(33, 150, 243);
text-decoration: none;
color: white;
border-radius: 4px;
font-size: 14px;
font-weight: 600;
min-width: 80px;
padding-top: 10px;
padding-bottom: 10px;
padding-left: 1rem;
padding-right: 1rem;
}

input.submit:hover {
background-color: rgb(72, 168, 245);
}

.result {
display: none;
}

.resultPassword {
display: initial;
}

.resultToken {
display: initial;
}
</style>
</head>
<body>
<img class="logo"
src="" />
<form method="POST" action="/auth">
<div class="message">{message}</div>
<form class="{formClass}" method="POST" action="{formAction}">
{form_fields}
<div class="message">{message}</div>
<div>
<input class="field" autocomplete="off" type="text" placeholder="User name" name="username" required autofocus />
</div>
<div>
<input class="field" type="password" placeholder="Password" name="password" required />
</div>
<div>
<input class="field" type="{repeatPasswordFieldType}" placeholder="Confirm Password" name="password_repeat" />
<input class="field" type="{newPasswordFieldType}" placeholder="New Password" name="new_password" />
</div>
<div>
<input class="field" type="{repeatPasswordFieldType}" placeholder="Confirm New Password" name="password_repeat" />
</div>
<div>
<input class="field" type="{tokenNameFieldType}" placeholder="Token Name" name="token_name" />
</div>
<div>
<input class="field" type="{tokenScopeFieldType}" placeholder="Token Scope (optional)" name="token_scope" />
</div>
<div>
<input class="submit" type="Submit" value="{buttonLabel}" />
</div>
</form>
<div class="result{resultClass}">
<a class="submit" type="button" href="/">Return Home</a>
</div>
</body>
</html>
Loading