⛏️
공부방
  • README.md
  • 프로젝트
    • ft_transcendence
      • 설계
        • 0. 프론트 디자인
        • 1. BackboneJS 뷰 객체
        • 2. API 설계
        • 3. 레일즈 라우팅 구현
        • 4. DB 설계
        • 5. 채널 설계
    • slab-saver
    • react-payment
  • 공부
    • HTML, CSS
      • GRID
      • emmet
      • position
      • CSS Unit
        • 단위 정리
        • 기준을 정해보자
        • em의 정확한 기준은 뭐야?
      • flex
      • NAVBAR 실습
      • 유튜브 화면 만들어보기
    • SQL
      • 이론
        • 1강 데이터베이스
        • 2강 다양한 데이터 베이스
        • 3강 데이터베이스 서버
      • 명령어
        • DB 관리
        • TABLE 관리
        • Constraints
        • SQL 명령어 - 1
        • SQL 명령어 - 2
        • SQL 명령어 - 3
        • SQL 명령어 - 4
        • SQL 명령어 - 5
    • Ruby
      • 루비 객체와 클래스
      • 곡괭이
        • Chapter2. Ruby.new
        • Chapter3. 클래스, 객체, 변수
        • Chapter4. 컨테이너, 블록, 반복자
        • Chapter5. 기능 공유하기
        • Chapter6. 표준 타입
        • Chapter8. 메서드 파헤치기
    • Python
      • 유용한 링크
    • RubyOnRails
      • 아직 정리하지 못한 것들
        • RSPEC 을 이용한 테스트 완전 자동화
        • 레일즈 이니셜라이징 과정
        • 액션케이블 구체적으로 정리하기
        • 웹팩으로 자바스크립트 모듈 관리하기
      • ACTIVE JOB
        • 액티브잡의 기본
        • 실전! 액티브 잡을 이용한 스케쥴링
        • 서버를 껏다 키면 스케쥴링 된 이벤트가 사라진다!
      • ACTION CABLE
        • 액션케이블 Consumer를 이용해서 문제 해결
        • 액션케이블 연결 순서
      • ACTIVE STORAGE
      • 모델
        • validation
          • seeds 데이터 validation 스킵
          • validation 검사가 save, update, create 모든 경우에 일어난다
          • validator 클래스
          • 커스텀Validation
          • validates format(정규표현식)
        • 액티브레코드 find의 다양한 활용
        • 한 레코드에 동시 접속 막자!! with_lock
        • 레일즈 where 사용법
        • 레일즈에서 모델 관련 이슈
        • 모델이름바꿀때명심할것
        • 모델의 includes 메서드
        • 연관 모델을 다른 이름으로 설정하고 가져오기
      • 기본 상식
        • form으로 전달되는 params를 분석해보자
        • StrongParameter 쿼리 배열 받기
        • view helper로 디버깅 하는 방법
        • css 파일을 수정했는데 적용이 안된다?
        • StrongParameter 일반데이터와 객체 데이터 한번에 받기
        • wrap-parameter body가 두 번씩 날라오는 이유
        • 컬렉션 map에서 요소 스킵하는법
        • CASE를 이용해서 정렬(일반적인 정렬 X)
        • 문자열(정규표현식)
        • TIME ZONE 설정하기
        • 커스텀exception
      • RSPEC으로 모델 테스트하기
      • 한 눈에 읽는 루비 온 레일즈
      • Perfect RubyOnRails
        • Chapter1. 소개
        • Chapter2. RubyOnRails 기본
        • Chapter3. 스캐폴딩
        • Chapter7. 라우팅
        • Chapter8. 테스트
    • Javascript
      • var, let, const 차이
      • 브라우저 동작 원리
      • 디바운싱과 쓰로틀링
      • Tagged Template Literal(styled-components)
      • IntersectionObserver 를 사용해서 스크롤 이벤트의 부하 줄여주기
      • EVENT LOOP
        • 자바스크립트에서 어떻게 비동기적인 실행이 가능한걸까?
        • 이벤트 루프의 동작
        • setTimeout이 실행되면 어떤 동작이 일어날까?
        • 블록은 실행이 보장된다
        • 콜스택에 있는 블록이 보장된다는 점을 이용해서 브라우저 죽이기
        • setTimeout 무한반복으로 브라우저는 죽을까?
        • Promise 무한반복으로는 브라우저를 죽일 수 있을까?
        • RAF는 그럼 뭐야?
      • forEach는 반복도중 멈출 방법이 throw 밖에 없다!
      • 임시
        • 정리할 것 목록
          • 자바스크립트 기본 문법
        • 이벤트 임시 정리
      • 유용한 링크
      • arrow function 을 이용한 bind 이슈 해결
      • preventDefault - passive
      • CRITICAL-RENDERING-PATH
      • setInterval에 클로져 개념 사용하기
      • 오디오 문제 이슈
      • 자바스크립트의 식과 문
        • 식과 문이란 무엇인가...
        • 식
          • 1. 기본값과 래퍼객체
          • 2. 참조값과 가비지컬렉팅
        • 식을 조금 더 자세히 알아보자
      • prototype, [[Prototype]] 차이
      • export, import 학습
      • ESlint
      • 아주아주기본
        • Chatper1. 기본
        • Chapter2. 타입
        • Chapter3. 연산자
        • Chapter4. 제어문
        • Chapter5. 배열
        • Chapter6. 함수
        • Chapter7-1. 객체
        • Chapter7-2. 객체
        • Chapter8. 표준객체
        • Chapter9. DOM
      • 이벤트 위임
      • 이벤트가 버블링 되서 root 까지 가다보면... 부모의 부모의 ... 모든 click 이벤트를 발동시키는거 아니야?
      • classList
    • BackboneJS
      • Backbone Model 프로토타입에 메서드 구현하기
      • BackboneJS의 각 요소의 역할과 책임을 확실히 이해하자
      • Window 이벤트를 listenTo로 감시하기
      • 뷰 자신이 자신을 지워야 할 때를 감지하려면 어떻게 해야하는가?
      • 백본 VIEW의 remove와 jquery의 remove 는 다르다!
      • 백본 컬렉션 URL에 쿼리 붙이기
      • index.html.erb와 BackboneJS의 결합
      • 백본 모델과 컬렉션에서 fetch 를 통해 JSON 가져오기!
      • 모델은 urlRoot, 컬렉션은 url
      • ISSUE
      • Absolute Beginner
        • Part1
        • Part2
        • Part3
        • Part4
    • 문제풀이
      • 01. 유효한 팰린드롬(leetcode: 125)
      • 02. 문자열 뒤집기(leetcode: 344)
      • 03. 로그파일 재정렬(leetcode 937)
      • 04. 가장 흔한 단어(leetcode: 819)
      • 05. 그룹 애너그램(leetcode: 49)
      • 06. 가장 긴 팰린드롬 문자열(leetcode: 5)
      • 07. 두 수의 합(leetcode: 1)
      • 08. 빗물 트래핑
      • 09. 세 수의 합(leetcode: 15)
    • BlackCoffeeStudy
      • level1
        • 1주차
    • express
      • Untitled
      • 구글 애널리틱스 연결하기
      • passport를 활용한 로그인
      • express-init 명령어 사용
      • ec2와 DBeaver
      • mariadb 설치
      • sequelize 설치 및 사용법
        • sequelize 설치
        • sequelize-cli 사용법
        • 모델 간 연관관계 맺기
        • Hook 사용하기
      • express-ejs-layout 활용하기
      • Bootstrap
      • npm install로 설치한 모듈 ejs에서 사용하기
      • 미들웨어
    • cypress
      • window.alert 테스트는 어떻게 하지?
      • 상수를 어디에 저장할건가?
      • before()와 beforeEach()
    • aws
      • aws로 프로젝트를 배포해보자!
      • nginx로 리버스프록시 서버를 만들자
      • github actions 로 푸쉬되면 자동으로 업데이트 하는 기능 만들어보기
    • react
      • Drag & Drop 를 이용해서 리스트 요소 순서 바꾸기
      • CRA에서 CRACO 사용하지 않고 절대경로 import(NODE_PATH)
      • useEffect내에서 state의 dependency 문제
      • IntersectionObserverAPI로 무한스크롤 구현
      • react-testing-library
        • 기본
        • react-router-dom 에서의 에러
        • event 발생시키기
        • Integration testing하기
        • async하게 렌더링 되는 요소 잡기
        • Mocking 하기
      • CRA로 만든 앱에서 절대경로로 import 해오기(alias하기)
      • 커스텀 훅 만들기
    • 타입스크립트
      • 조건부타입 (Conditional types)
      • Generics
      • Keyof 타입 오퍼레이터
      • Indexed Access Types
      • 타입 챌린지
        • easy
          • 00. Awaited
          • 01. Concat
          • 02. Exclude
          • 03. First Of Array
          • 04. If
          • 05. Includes
          • 06. Pick
          • 07. Readonly
          • 08. Length
          • 09. Tuple to Object
        • mediun
          • 01. Absolute
    • Firebase
      • 파이어스토어 규칙
    • 기타
      • 협업 프로세스
      • UUID
      • 구글애널리틱스 설치하기
      • 드림코딩 강의
        • 포트폴리오
          • CSS
            • nth-child
            • CSS 팁
          • 자바스크립트
            • 1. 스크롤에 따른 navbar 의 색 변경하기
            • 2. navbar 버튼을 누르면 해당 페이지로 스크롤링 되게 만들자
            • 3. 스크롤 다운 하면 arrow-up 버튼 나오게 하기
            • 4. project 필터링 구현
            • 5. project 필터링에 transition 효과 넣기
      • GIT
        • 기본 사용법 정리
        • git remote update - remote 브랜치 가져오기
  • 기타
    • 이것저것
      • 독서
        • 클린코드
          • Chapter0. 나는 왜 클린코드 책을 읽는가?
          • Chapter1. 클린코드
          • Chapter2. 의미있는 이름
          • Chapter3. 함수
          • Chapter4. 주석
          • Chapter5. 형식 맞추기
      • 용어
      • IDE
        • RubyMine
          • 실전이 중요!
          • 1. Editor Basic
          • 2. Navigation
          • 3. Completion
          • 4. Refactoring
          • 5. Code Assistance
      • MAC에서 살아남기
        • Alfred - Spotlight 업그레이드
        • Vimium
        • BetterTouchTool - 트랙패드
        • 구름 입력기 - ESC, `
        • Spectacle - 화면 분할
    • 원티드 프리온보딩
      • 1주차
        • 월요일
        • 목요일
      • 2주차
      • 3주차
      • 4주차
      • 5주차
      • 6주차
    • 일기장
      • 2020
        • December
          • 20201208(화)
          • 20201209(수)
          • 20201210(목)
          • 20201211(금)
          • 20201214(월)
          • 20201215(화)
          • 20201216(수)
          • 20201217(목)
          • 20201218(금)
          • 20201219(토)
          • 20201221(월)
          • 20201222(화)
          • 20201223(수)
          • 20201224(목)
          • 20201226(토)
          • 20201228(월)
          • 20201229(화)
          • 20201230(수)
          • 20201231(목)
      • 2021
        • January
          • 20210101(금)
          • 20210102(토)
          • 20210105(화)
          • 20210106(수)
          • 20210107(목)
          • 20210108(금)
          • 20210109(토)
          • 20210112(화)
          • 20210113(수)
          • 20210114(목)
          • 20210115(금)
          • 20210117(일)
          • 20210118(월)
          • 20210119(화)
          • 20210120(수)
          • 20210121(목)
          • 20210125(월)
          • 20210126(화)
          • 20210127(수)
          • 20210128(목)
          • 20210129(금)
        • February
          • 20210201(월)
          • 20210202(화)
          • 20210203(수)
          • 20210204(목)
          • 20210205(금)
          • 20210207(일)
          • 20210208(월)
          • 20210209(화)
          • 20210217(수)
          • 20210218(목)
          • 20210219(금)
          • 20210220(토)
          • 20210222(월)
          • 20210223(화)
          • 20210224(수)
          • 20210226(금)
          • 20210228(일)
        • March
          • 20210302(화)
          • 20210303(수)
          • 20210304(목)
          • 20210305(금)
          • 20210306(토)
          • 20210308(월)
          • 20210309(화)
          • 20210310(수)
          • 20210311(목)
          • 20210312(금)
          • 20210313(토)
          • 20210315(월)
          • 20210316(화)
          • 20210317(수)
          • 20210318(목)
          • 20210319(금)
          • 20210322(월)
          • 20210323(화)
          • 20210324(수)
          • 20210325(목)
          • 20210327(토)
          • 20210329(월)
          • 20210330(화)
          • 20210331(수)
        • April
          • 20210406(화)
          • 20210407(수)
          • 20210408(목)
          • 20210409(금)
          • 20210410(토)
          • 20210412(월)
          • 20210413(화)
          • 20210414(수)
          • 20210415(목)
          • 20210416(금)
          • 20210417(토)
          • 20210419(월)
          • 20210420(화)
          • 20210421(수)
          • 20210422(목)
        • July
          • 20210728(수)
Powered by GitBook
On this page
  • 7.1 RESTful 인터페이스
  • 7.1.1 RESTful 인터페이스의 정의
  • 7.1.2 하나의 리소스 정의
  • 7.1.3 라우트 정의 확인하는 법
  • 7.2 RESTful 인퍼테이스의 사용자 정의화
  • 7.2.1 라우트 매개 변수 제약 조건
  • 7.2.2 복잡한 제약 조건 설정
  • 7.2.3 format 매개 변수 제거
  • 7.2.4 컨트롤러 클래스와 URL 헬퍼의 이름 수정
  • 7.2.5 모듈 내부의 컨트롤러를 맵핑
  • 7.2.6 RESTful 액션 추가
  • 7.2.7 RESTful 액션 무효화
  • 7.2.8 계층 구조를 가진 리소스 표현
  • 7.2.9 리소스의 얕은 중첩 표현
  • 7.2.10 라우 정의 재이용
  • 7.3 RESTful 하지 않은 라우트 정의의 기본
  • 7.3.1 RESTful 하지 않은 라우트 정의의 기본

Was this helpful?

  1. 공부
  2. RubyOnRails
  3. Perfect RubyOnRails

Chapter7. 라우팅

라우팅은 요청 URL에 따라 응답 처리 대상(액션)을 결정하는 것 또는 그 구조 자체를 말한다.

7.1 RESTful 인터페이스

RESTful 인터페이스를 잘 사용하면 통일감 있고 의미가 명확한 URL 설계가 가능하다.

  • RESTful 인터페이스는 REST의 측징을 가진 라우트를 말한다.

  • REST는 네트워크의 모든 컨텐츠(리소스)를 URL로 표현한다. 그리고 이러한 URL(리소스)에 HTTP 메서드 (GET-추출, POST-생성, PATCH-변경, DELETE-제거) 등의 방법으로 접근한다.

  • 즉, REST란 무엇(리소스)을 어떻게(HTTP 메서드)할지 표현하는 것이다.

  • Rails 는 원칙적으로 RESTful 인터페이스를 기반으로 라우트를 설계한다.

    • 따라서 Rails에서 제공하는 fomr_with, link_to 등의 뷰 헤퍼는 RESTful 인터페이스를 전제로 설계 되었다.

7.1.1 RESTful 인터페이스의 정의

  • RESTful 인터페이스를 정의할 때는 routes.rb 에서 resources 메서드를 호출한다.

  • 리소스는 CRUD 할 수 있는 정보라고 생각하면 좋다. 아래처럼 routes.rb 파일에 resources :name 형식으로 입력하면 URL 로 액션이 매핑된다.

    • 즉, resources 메서드 한방으로 RESTful 인터페이스가 정의 된다는 것이다.

    Railbook::Application.routes.draw do
      resources :users
      ...
      end
  • HTTP Verb

    Path

    Controller#Action

    Used for

    GET

    /users

    users#index

    display a list of all users

    GET

    /users/new

    users#new

    return an HTML form for creating a new user

    POST

    /users

    users#create

    create a new user

    GET

    /users/:id

    users#show

    display a specific user

    GET

    /users/:id/edit

    users#edit

    return an HTML form for editing a user

    PATCH/PUT

    /users/:id

    users#update

    update a specific user

    DELETE

    /users/:id

    users#destroy

    delete a specific user

    resources 메서드는 뷰 헬퍼(link_to)등 에서 사용할 수 있는 URL 헬퍼 도 자동으로 생성한다. 이러한 헬퍼를 사용함으로써 링크를 보다 직관적으로 사용할 수 있는 것은 물론 라우트 정의에 의존하지 않을 수도 있다.

    헬퍼 이름(_path)

    헬퍼 이름(_URL)

    리턴 값(경로))

    users_path

    users_url

    /users

    user_path(id)

    user_url(id)

    /users/:id

    new_user_path

    new_user_url

    /users/new

    edit_user_path(id)

    edit_user_url(id)

    /users/:id/edit

  • xxx_path 와 xxx_url 의 차이는

    • xxx_path 는 상대경로

    • xxx_url 는 절대 경로

    • xxx_url 는 format 을 인자로 받을 수 있다. 따라서 respond_to 처럼 format 에 따라 각기 다른 응답을 해야할 때에는 xxx_url 을 사용한다.

7.1.2 하나의 리소스 정의

resource 메서

  • resources 는 복수형이다. 즉 여러 개의 리소스를 관리하는 RESTful 인터페이스를 생성한다.

  • 반면 resource는 하나의 리소스를 관리하는 RESTful 인터페이스를 생성한다.

  • 예를 들어 하나의 리소스는 애플리케이션의 설정 정보 등의 리소스를 말한다. 즉 애플리케이션 설정은 당연히 그 애플리케이션에서 유일하므로 /config/15 가 아니라 /config 의 형태로 URL에 접근해야 할 것이다.

  • resource config의 결과는 아래와 같다.

    URL

    액션

    HTTP 메서드

    역할

    /config(:format)

    show

    GET

    설정 정보 화면 표시

    /config/new(:format)

    new

    GET

    신규 설정 등록 화면 표시

    /config(:format)

    create

    POST

    등록 화면에서 입력을 받아 등록 처리

    /config/edit(:format)

    edit

    GET

    기존 설정 수정 화면 표시

    /config(:format)

    update

    PATCH/PUT

    수정 화면에서 입력을 받아 수정 처리

    /config(:format)

    destroy

    DELETE

    지정된 설정 정보를 제거 처리

  • resources 메서드와 거의 비슷하지만 index 액션이 정의 되지 않고, :id를 인자로 받지 않는다.

  • 주의해야할 것은 resource 메서드와 매칭되는 컨트롤러는 반드시 복수형이어야 한다.

    • 즉 config의 컨트롤러 이름은 ConfigsController처럼 복수형이 되어야만 한다.

  • URL 헬퍼도 생성이된다.

    PATH

    URL

    경로

    config_path

    config_url

    /config

    new_config_path

    new_config_path

    /config/new

    edit_config_path

    edit_config_path

    /config/edit

7.1.3 라우트 정의 확인하는 법

  1. 터미널에 rails routes 입력하기

  2. 브라우저에 localhost:3000/rails/info/routes로 접근하기

7.2 RESTful 인퍼테이스의 사용자 정의화

  • Rails에서는 resources와 resource메서드를 사용해 정형적인 라우트를 자동으로 생성할 수 있다. 이러한 간단함이 RESTful 인터페이스의 장점이다.

  • 추가로 옵션을 사용해 더욱 다채롭게 사용할 수 있다.

7.2.1 라우트 매개 변수 제약 조건

constraints 옵션

  • resources 메서드로 자동 생성되는 URL 에는 :id 라는 이름의 라우트 매개 변수가 포함된다.

  • 이 :id 라우트 매개 변수에 제약 조건을 걸 수 있다.

  • 아래와 같은 명령을 입력하면 id의 값은 숫자 1자리 또는 2자리로 제한된다.

resources :books, constraints: { id: /[0-9]{1,2}/}
  • 다만 라우트 매개 변수와 관련된 유효성 검사는 모델에서 하는 것이 기본 이다.

7.2.2 복잡한 제약 조건 설정

제약 클래스를 정의 하자

  • 위와 같이 간단한 제약은 resources의 옵션에서 바로 넣을 수 있지만, 복잡한 제약 조건을 걸기 위해서는 클래스를 만들어야 한다.

  • 순서는 아래와 같다. 아래는 9시부터 18시 까지만 라우팅이 가능한 제약을 만드는 예시.

    1. /models 폴더에 TimConstraint.rb 를 만든다.

    2. 파일이름과 같은 클래스 를 만들고 match? 메서드를 만든다.

    3. class TimeConstraint
        def matches?(request)
          ... 생략
        end
      end
    4. routes.rb의 resources 에 제약을 넣으면 된다.

    5. resources :books, constraints: TimeConstraint.new

7.2.3 format 매개 변수 제거

Format 옵션

  • resources 메서드로 RESTful 인터페이스를 만들고, 터미널에 rails routes 를 입력하면 라우팅 정보가 나온다.

  • 이때 경로를 보면 뒤에 (:format) 이 붙어있는게 보인다. 이게 뭐냐면 클라이언트가 html 파일이 아닌 .json 이나 .xml 등과 같은 포맷으로 요청을 날릴 때 이 요청 또한 소화하기 위해 처리해 놓은것이다.

  • 만약 서버가 이런 포맷을 원하지 않는다? 그러면 옵션을 이용해서 없애자.

  • resources :books, format: false
  • 이후 rails routes | grep books 를 입력해서 :format이 사라진걸 확인하자

7.2.4 컨트롤러 클래스와 URL 헬퍼의 이름 수정

controller, as 옵션

  • resources 메서드는 기본적으로 지정된 리소스 이름을 기반으로 대응되는 컨트롤러를 결정하고 URL 헬퍼를 생성한다.

  • 만약 기본으로 지정된 이름이 아닌 다른 이름의 컨트롤러를 매핑하고 싶다면 controller 옵션을 사용하자

    • resources :users, controller
  • URL 헬퍼의 이름만 바꾸고 싶다? 그러면 ad 옵션을 사용하자

    • resources :users as members
    • 기존의 users_path 는 members_path, user_path는 member_path 등으로 변경된다.

7.2.5 모듈 내부의 컨트롤러를 맵핑

namespace, scope 블록

  • 컨트롤러 클래스의 수가 많아지면 모듈을 사용해 컨트롤러를 폴더로 정리해야 한다.

  • 이때는 아래와 같이 컨트롤러 클래스를 생성한다.

    rails generate controller Admin::Books
  • 이렇게 하면 Admin::BooksConttoller가 controllers/admin 폴더에 books_controller.rb 라는 이름으로 생성된다.

  • 이렇게 모듈을 사용한 컨트롤러 클래스에 RESTful 인터페이스를 정의하려면 namespace 블록을 사용한다.

  • namespace :admin do
      resources :books
    end
  • 이렇게 하고 rails routes 를 확인해보면 books 앞에 admin이 붙은걸 확인할 수 있음.

    • 즉 너무 길어진다.

  • 만약 길어지는게 싫다면 scope 블록을 사용하면 된다.

  • scope midules: :admin do
      resources :books
    end

7.2.6 RESTful 액션 추가

collection과 member 블록

  • resources로 자동 생성되는 라우투는 CRUD에 한정되어 있다.

  • 나만의 라우트를 추가 하고 싶다면 collection 또는 member 블록을 사용하면 된다.

  • collection

    • 이 블록은 이름만 봐도 복수로 표현될 라우트를 추가하는 것.

    • index 와 마찬가지로 :id 라우트 매개 변수가 필요 없다!

  • member

    • 이 블록은 하나의 리소스를 지정해야 한다.

    • 즉 :id 매개변수가 필요하다!

  • resources :reviews do
      collection do
        get :unapproval
      end
      member do
        get :draft
      end
    end
    ---결과---
    unapproval_reviews GET /reviews/unapproval(:format)
    draft_review            GET /reviews/:id/draft(:format)

7.2.7 RESTful 액션 무효화

only, except 옵션

  • resources :users, except: [ :show, :destroy ]
  • resources :users, only: [ :index, :new, :create, :edit, :update ]

7.2.8 계층 구조를 가진 리소스 표현

resources 메서드 중첩

  • 리소스들이 애플리케이션 내부에서 계층 구조를 갖는 경우가 있다.

  • 예를 들면 book 리소스에는 reviews 리소스가 포함된다.

    • book 은 review를 has_many

    • review는 book에 belongs_to

  • 이러한 모델 관계는 URL 로도 표현해주는 것이 직관적이다. 예를 들면 book_id: 1의 reviews 를 보고 싶을 때는 /books/1/reviews 처럼 해주는게 우리가 생각하는 일반적인 모습과 같을 것이다.

  • 이 때 사용하는게 resources의 중첩 구조다. 사용법과 그로인해 생기는 결과는 아래와 같다.

  • resources :books do
      resources :reviews
    end
  • 먼저 xxx_path 의 이름이 기존과 다르게 book_reviews 와 같이 변경됐고, path 에 들어가는 인자도 :id 하나가 아닌 book_id 와 :id 두개가 되었다.

  • 이렇게 만든 상태로 서버에 접근하면 서버 에러가 발생한다. 에러가 발생하는 원인은 간단하다.

    • 먼저 reviews 의 html.erb 파일들에서 사용하고 있던 URL 헬퍼 들이 수정되지 않은 상태다.

      • reviews_path 였다면 book_reviews로 바꿔야 한다.

  • 이런 헬퍼들에 대한 인자가 2개로 바꼈다. 기존에는 review의 :id 하나면 됐지만 이제는 book_id 까지 넣어줘야 한다.

  • 예를 들면 reviews/index.html.erb 파일을 아래처럼 수정해야한다.

    --- 변경 전 ---
    <td><%= link_to 'Show', review %></td>
    --- 변경 후 ---
    <td><%= link_to 'Show', book_review_path(book_id: review.book_id, id: review.id) %></td>

7.2.9 리소스의 얕은 중첩 표현

shallow 표현

  • resources로 중첩표현을 하면 위의 사진처럼 특정 review에 접근하기 위해서는 book 과 관련된 처리까지 해야한다.

  • shallow 옵션을 사용하면 이 번거로움을 해결할 수 있다.

  • resources :books do
      resources :reviews, shallow: true
    end
  • :id 매개 변수를 받지 않는 index, new, create 액션은 그대로지만

  • :id 매개 변수를 받는 edit, show, update ... 등은 book_id 가 사라졌음을 확인할 수 있다.

  • shallow를 하드코딩 하는 방법도 있다.

    resources :books do
      resources :reviews, only: [ :index, :new, :create ]
    end
    resources :reviews, except: [ :index, :new, :create]

7.2.10 라우 정의 재이용

concern 메서드와 concerns 옵션

  • concern 메서드를 사용하면 공통되는 내용을 여러 개의 라우트 정의에 넣을 수 있다.

  • resources :reviews do
      get :unapproval, on: :collection
      get :draft, on: :member
    end
    
    resources :users do
      get :unapproval, on: :collection
      get :draft, on: :member
    end
  • 위와 같은 라우트 정의는 블록 내부의 내용이 같으므로 concern 메서드로 분리하자.

    concern :additional do
      get :unapproval, on: :collection
      get :draft. on: :member
    end
    
    resources :reviews, concern: :additional
    resources :users, concerns: :additional

7.3 RESTful 하지 않은 라우트 정의의 기본

  • Rails는 기본적으로 RESTful 인터페이스로 구성된다. 하지만 모든 것을 REST의 사상에 맞춰 만들 수 있는 것은 아니다.

  • 이럴 때는 무리해서 RESTful 인터페이스를 적용하려 들지 말고 간단하게 RESTful 하지 않은 라우트 정의를 사용해도 된다.

    • 무조건 RESTful 인터페이스에 따라 만드는 것이 좋은것은 아니다!

7.3.1 RESTful 하지 않은 라우트 정의의 기본

match 메서드

# match pattern, via: verb [,opts]

match ':controller(/:action(/:id))', via: [ :get, :post, :patch ]
  • 위와 같은 형태로 패턴을 만들면 아래와 같은 패턴이 매치된다.

    • /books

    • /hello/list

    • /blog/details/123

    • blog/details/123.json

  • 괄호로 감싸진 부분은 생략 가능함을 나타낸다.

  • via 옵션은 라우트에 허용할 HTTP 메서드 를 지정한다.

PreviousChapter3. 스캐폴딩NextChapter8. 테스트

Last updated 4 years ago

Was this helpful?

계층구조에서의 reviews 라우팅 테이블
shallow 옵션을 적용했을 때의 reviews 라우팅 테이블