Redditアプリでログインから登録を分離する

Redditアプリでの登録とログインの分離

1. 概要

このチュートリアルでは–we’ll replace the Reddit backed OAuth2 authentication process with a simpler, form-based login

ログイン後もRedditを最大the applicationまでフックできますが、メインのログインフローを駆動するためにRedditを使用することはありません。

2. 基本的なユーザー登録

まず、古い認証フローを置き換えましょう。

2.1. Userエンティティ

Userエンティティにいくつかの変更を加えます。usernameを一意にし、passwordフィールドを追加します(一時的):

@Entity
public class User {
    ...

    @Column(nullable = false, unique = true)
    private String username;

    private String password;

    ...
}

2.2. 新規ユーザーの登録

次へ–バックエンドに新しいユーザーを登録する方法を見てみましょう。

@Controller
@RequestMapping(value = "/user")
public class UserController {

    @Autowired
    private UserService service;

    @RequestMapping(value = "/register", method = RequestMethod.POST)
    @ResponseStatus(HttpStatus.OK)
    public void register(
      @RequestParam("username") String username,
      @RequestParam("email") String email,
      @RequestParam("password") String password)
    {
        service.registerNewUser(username, email, password);
    }
}

明らかに、これはユーザーにとって基本的な作成操作です。機能はありません。

the actual implementation, in the service layerは次のとおりです。

@Service
public class UserService {
    @Autowired
    private UserRepository userRepository;

    @Autowired
    private PreferenceRepository preferenceReopsitory;

    @Autowired
    private PasswordEncoder passwordEncoder;

    @Override
    public void registerNewUser(String username, String email, String password) {
        User existingUser = userRepository.findByUsername(username);
        if (existingUser != null) {
            throw new UsernameAlreadyExistsException("Username already exists");
        }

        User user = new User();
        user.setUsername(username);
        user.setPassword(passwordEncoder.encode(password));
        Preference pref = new Preference();
        pref.setTimezone(TimeZone.getDefault().getID());
        pref.setEmail(email);
        preferenceReopsitory.save(pref);
        user.setPreference(pref);
        userRepository.save(user);
    }
}

2.3. 例外への対処

そして、単純なUserAlreadyExistsException

public class UsernameAlreadyExistsException extends RuntimeException {

    public UsernameAlreadyExistsException(String message) {
        super(message);
    }
    public UsernameAlreadyExistsException(String message, Throwable cause) {
        super(message, cause);
    }
}

例外はin the main exception handler of the applicationで処理されます。

@ExceptionHandler({ UsernameAlreadyExistsException.class })
public ResponseEntity
  handleUsernameAlreadyExists(RuntimeException ex, WebRequest request) {
    logger.error("400 Status Code", ex);
    String bodyOfResponse = ex.getLocalizedMessage();
    return new
      ResponseEntity(bodyOfResponse, new HttpHeaders(), HttpStatus.BAD_REQUEST);
}



2.4. 簡単な登録ページ

最後に–単純なフロントエンドsignup.html

これは完全に成熟した登録プロセスではなく、非常に迅速なフローであることに再度言及する価値があります。 完全な登録フローについては、例としてここでthe main registration seriesを確認できます。

3. 新しいログインページ

new and simple login pageは次のとおりです。

Invalid username or password
Sign up

4. セキュリティ構成

では、the new security configurationを見てみましょう。

@Configuration
@EnableWebSecurity
@ComponentScan({ "org.example.security" })
public class SecurityConfig extends WebSecurityConfigurerAdapter {

    @Autowired
    private MyUserDetailsService userDetailsService;

    @Override
    protected void configure(AuthenticationManagerBuilder auth) throws Exception {
        auth.userDetailsService(userDetailsService).passwordEncoder(encoder());
    }

    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http
            ...
            .formLogin()
            .loginPage("/")
            .loginProcessingUrl("/j_spring_security_check")
            .defaultSuccessUrl("/home")
            .failureUrl("/?error=true")
            .usernameParameter("username")
            .passwordParameter("password")
            ...
    }

    @Bean
    public PasswordEncoder encoder() {
        return new BCryptPasswordEncoder(11);
    }
}

ほとんどのことは非常に簡単なので、ここでは詳しく説明しません。

そして、これがカスタムUserDetailsServiceです。

@Service
public class MyUserDetailsService implements UserDetailsService {

    @Autowired
    private UserRepository userRepository;

    @Override
    public UserDetails loadUserByUsername(String username) {
        User user = userRepository.findByUsername(username);
        if (user == null) {
            throw new UsernameNotFoundException(username);
        }
        return new UserPrincipal(user);
    }
}

そして、これがUserDetailsを実装するカスタムPrincipalUserPrincipal”です:

public class UserPrincipal implements UserDetails {

    private User user;

    public UserPrincipal(User user) {
        super();
        this.user = user;
    }

    @Override
    public String getUsername() {
        return user.getUsername();
    }

    @Override
    public String getPassword() {
        return user.getPassword();
    }

    @Override
    public Collection getAuthorities() {
        return Arrays.asList(new SimpleGrantedAuthority("ROLE_USER"));
    }

    @Override
    public boolean isAccountNonExpired() {
        return true;
    }

    @Override
    public boolean isAccountNonLocked() {
        return true;
    }

    @Override
    public boolean isCredentialsNonExpired() {
        return true;
    }

    @Override
    public boolean isEnabled() {
        return true;
    }
}

注:Spring SecurityのデフォルトのUserの代わりに、カスタムPrincipal "UserPrincipal”を使用しました。

5. Redditを認証する

認証フローをRedditに依存しなくなったので、ログイン後にenable users to connect their accounts to Redditする必要があります。

まず、古いRedditログインロジックを変更する必要があります。

@RequestMapping("/redditLogin")
public String redditLogin() {
    OAuth2AccessToken token = redditTemplate.getAccessToken();
    service.connectReddit(redditTemplate.needsCaptcha(), token);
    return "redirect:home";
}

そして実際の実装–connectReddit()メソッド:

@Override
public void connectReddit(boolean needsCaptcha, OAuth2AccessToken token) {
    UserPrincipal userPrincipal = (UserPrincipal)
      SecurityContextHolder.getContext().getAuthentication().getPrincipal();
    User currentUser = userPrincipal.getUser();
    currentUser.setNeedCaptcha(needsCaptcha);
    currentUser.setAccessToken(token.getValue());
    currentUser.setRefreshToken(token.getRefreshToken().getValue());
    currentUser.setTokenExpiration(token.getExpiration());
    userRepository.save(currentUser);
}

redditLogin()ロジックを使用して、ユーザーのAccessTokenを取得することにより、システム内のユーザーのアカウントをRedditアカウントに接続する方法に注意してください。

フロントエンドに関しては–それは非常に簡単です:

また、投稿を送信する前に、ユーザーがアカウントをRedditに接続していることを確認する必要もあります。

@RequestMapping("/post")
public String showSubmissionForm(Model model) {
    if (getCurrentUser().getAccessToken() == null) {
        model.addAttribute("msg", "Sorry, You did not connect your account to Reddit yet");
        return "submissionResponse";
    }
    ...
}

6. 結論

小さなredditアプリは間違いなく前進しています。

Redditによって完全にサポートされている古い認証フローは、いくつかの問題を引き起こしていました。 これで、we have a clean and simple form-based loginは、バックエンドでRedditAPIに接続できます。

良いもの。