4 changed files with 216 additions and 1 deletions
@ -0,0 +1,64 @@ |
|||
package app.controllers; |
|||
|
|||
import app.services.SteamSignIn; |
|||
import app.utils.SaltedCookie; |
|||
import jakarta.servlet.http.Cookie; |
|||
import jakarta.servlet.http.HttpServletResponse; |
|||
import org.springframework.beans.factory.annotation.Autowired; |
|||
import org.springframework.http.HttpStatus; |
|||
import org.springframework.http.ResponseEntity; |
|||
import org.springframework.web.bind.annotation.GetMapping; |
|||
import org.springframework.web.bind.annotation.RequestMapping; |
|||
import org.springframework.web.bind.annotation.RequestParam; |
|||
import org.springframework.web.bind.annotation.RestController; |
|||
import java.util.Map; |
|||
|
|||
@RestController |
|||
@RequestMapping("/api/auth") |
|||
public class AuthController { |
|||
private SteamSignIn steamSignIn; |
|||
private SaltedCookie saltedCookie; |
|||
|
|||
@Autowired |
|||
public AuthController(SteamSignIn steamSignIn, SaltedCookie saltedCookie){ |
|||
this.steamSignIn = steamSignIn; |
|||
this.saltedCookie = saltedCookie; |
|||
} |
|||
|
|||
@GetMapping("login") |
|||
public ResponseEntity<Void> Login(){ |
|||
return steamSignIn.ConstructURLAndRedirect(); |
|||
} |
|||
|
|||
@GetMapping("logout") |
|||
public ResponseEntity<?> Logout(HttpServletResponse response){ |
|||
Cookie cookie_steam64 = new Cookie("steam64",""); |
|||
cookie_steam64.setMaxAge(0); |
|||
cookie_steam64.setPath("/"); |
|||
response.addCookie(cookie_steam64); |
|||
Cookie cookie_steam64_secured = new Cookie("steam64_secured", ""); |
|||
cookie_steam64_secured.setMaxAge(0); |
|||
cookie_steam64.setPath("/"); |
|||
response.addCookie(cookie_steam64_secured); |
|||
return ResponseEntity.ok().body("logout..."); |
|||
} |
|||
|
|||
@GetMapping("processlogin") |
|||
public ResponseEntity<?> ProcessLogin(@RequestParam Map<String, String> auth_result, HttpServletResponse response){ |
|||
System.out.println(auth_result); |
|||
Long steam64 = steamSignIn.ValidateResults(auth_result); |
|||
if(steam64 == null){ |
|||
return new ResponseEntity<>("returned steam is not valid",HttpStatus.FORBIDDEN); |
|||
} |
|||
|
|||
Cookie cookie_steam64 = new Cookie("steam64", steam64.toString()); |
|||
cookie_steam64.setPath("/"); |
|||
response.addCookie(cookie_steam64); |
|||
Cookie cookie_steam64_secured = new Cookie("steam64_secured", saltedCookie.Hashed(steam64.toString())); |
|||
cookie_steam64_secured.setPath("/"); |
|||
response.addCookie(cookie_steam64_secured); |
|||
|
|||
return ResponseEntity.ok() |
|||
.body("login successful"); |
|||
} |
|||
} |
@ -0,0 +1,123 @@ |
|||
package app.services; |
|||
|
|||
import org.springframework.beans.factory.annotation.Autowired; |
|||
import org.springframework.beans.factory.annotation.Value; |
|||
import org.springframework.http.*; |
|||
import org.springframework.stereotype.Service; |
|||
import org.springframework.web.client.RestTemplate; |
|||
import org.springframework.web.util.DefaultUriBuilderFactory; |
|||
import org.springframework.web.util.UriBuilder; |
|||
|
|||
import java.io.UnsupportedEncodingException; |
|||
import java.net.URI; |
|||
import java.net.URLDecoder; |
|||
import java.net.URLEncoder; |
|||
import java.nio.charset.StandardCharsets; |
|||
import java.util.Arrays; |
|||
import java.util.HashMap; |
|||
import java.util.List; |
|||
import java.util.Map; |
|||
import java.util.regex.Matcher; |
|||
import java.util.regex.Pattern; |
|||
import java.util.stream.Collectors; |
|||
|
|||
@Service |
|||
public class SteamSignIn { |
|||
private final String provider = "https://steamcommunity.com/openid/login?"; |
|||
|
|||
@Value("${backend.auth.response_redirect}") |
|||
private String responseURL; |
|||
|
|||
private RestTemplate restTemplate; |
|||
|
|||
private final Pattern refinedScripts = Pattern.compile("(?:http)"); |
|||
private final Pattern validAuth = Pattern.compile("is_valid:true"); |
|||
private final Pattern Check64ID = Pattern.compile("https://steamcommunity.com/openid/id/(\\d+)"); |
|||
|
|||
@Autowired |
|||
public void SteamSignIn(){ |
|||
this.restTemplate = new RestTemplate(); |
|||
|
|||
//http://www.chrispad.com/2019/04/disable-encoding-url-using-resttemplate.html
|
|||
//https://stackoverflow.com/questions/66164546/resttemplate-exchange-fail-on-get-call-but-works-on-curl
|
|||
//disable double encoding with rest
|
|||
DefaultUriBuilderFactory defaultUriBuilderFactory = new DefaultUriBuilderFactory(); |
|||
defaultUriBuilderFactory.setEncodingMode(DefaultUriBuilderFactory.EncodingMode.NONE); |
|||
this.restTemplate.setUriTemplateHandler(defaultUriBuilderFactory); |
|||
} |
|||
|
|||
private String encodeValue(String value){ |
|||
try { |
|||
return URLEncoder.encode(value, StandardCharsets.UTF_8.toString()); |
|||
} catch (UnsupportedEncodingException err){ |
|||
return ""; |
|||
} |
|||
} |
|||
|
|||
private String decodeValue(String value) { |
|||
try { |
|||
return URLDecoder.decode(value, StandardCharsets.UTF_8.toString()); |
|||
} catch (UnsupportedEncodingException err){ |
|||
return ""; |
|||
} |
|||
} |
|||
|
|||
public ResponseEntity<Void> ConstructURLAndRedirect(){ |
|||
Matcher result = refinedScripts.matcher(responseURL); |
|||
if (!result.find() || result.group(0).isEmpty()){ |
|||
responseURL = String.format("http://%s", responseURL); |
|||
} |
|||
|
|||
HashMap<String, String> authParameters = new HashMap<>(); |
|||
authParameters.put("openid.ns", "http://specs.openid.net/auth/2.0"); |
|||
authParameters.put("openid.mode", "checkid_setup"); |
|||
authParameters.put("openid.return_to", responseURL); |
|||
authParameters.put("openid.realm", responseURL); |
|||
authParameters.put("openid.identity", "http://specs.openid.net/auth/2.0/identifier_select"); |
|||
authParameters.put("openid.claimed_id", "http://specs.openid.net/auth/2.0/identifier_select"); |
|||
|
|||
String url = authParameters.keySet().stream() |
|||
.map(key -> key + "=" + encodeValue(authParameters.get(key))) |
|||
.collect(Collectors.joining("&", provider, "")); |
|||
|
|||
return ResponseEntity.status(HttpStatus.SEE_OTHER). |
|||
header("Content-Type", "application/x-www-form-urlencoded"). |
|||
location(URI.create(url)) |
|||
.build(); |
|||
} |
|||
|
|||
//@RequestParam Map<String, String> reqParam
|
|||
public Long ValidateResults(Map<String, String> results){ |
|||
Map<String, String> validationArgs = new HashMap<>(); |
|||
validationArgs.put("openid.assoc_handle", results.get("openid.assoc_handle")); |
|||
validationArgs.put("openid.signed", results.get("openid.signed")); |
|||
validationArgs.put("openid.sig", results.get("openid.sig")); |
|||
validationArgs.put("openid.ns", results.get("openid.ns")); |
|||
|
|||
List<String> signedArgs = Arrays.stream(results.get("openid.signed").split(",")).toList(); |
|||
for(String item: signedArgs){ |
|||
String itemArg = String.format("openid.%s", item); |
|||
if (!validationArgs.containsKey(results.get(itemArg))) { |
|||
validationArgs.put(itemArg, results.get(itemArg)); |
|||
} |
|||
} |
|||
validationArgs.put("openid.mode", "check_authentication"); |
|||
|
|||
String url = validationArgs.keySet().stream() |
|||
.map(key -> key + "=" + encodeValue(validationArgs.get(key))) |
|||
.collect(Collectors.joining("&", provider, "")); |
|||
|
|||
String responseData = restTemplate.getForObject(url, String.class); |
|||
|
|||
if (validAuth.matcher(responseData).find()) { |
|||
Matcher matcher = Check64ID.matcher(results.get("openid.claimed_id")); |
|||
if (!matcher.find() || !matcher.group(1).isEmpty()) { |
|||
return Long.valueOf(matcher.group(1)); |
|||
} else { |
|||
return null; |
|||
} |
|||
} else { |
|||
return null; |
|||
} |
|||
} |
|||
} |
@ -0,0 +1,23 @@ |
|||
package app.utils; |
|||
|
|||
import org.springframework.beans.factory.annotation.Value; |
|||
import org.springframework.stereotype.Component; |
|||
import org.springframework.util.DigestUtils; |
|||
|
|||
@Component |
|||
public class SaltedCookie { |
|||
@Value("${backend.auth.salt}") |
|||
private String salt; |
|||
|
|||
public String Hashed(String value) { |
|||
return DigestUtils.md5DigestAsHex(String.format("%s+%s", value, salt).getBytes()); |
|||
} |
|||
|
|||
public boolean Validate(String value, String hashed_value) { |
|||
return Hashed(value).equals(hashed_value); |
|||
} |
|||
|
|||
public boolean Validate(Long value, String hashed_value) { |
|||
return Validate(value.toString(), hashed_value); |
|||
} |
|||
} |
Loading…
Reference in new issue