마이크로서비스 아키텍처 – 설계와 구현

마이크로서비스 아키텍처 – 설계와 구현


마이크로서비스 아키텍처는 대규모 애플리케이션을 작은 독립된 서비스로 나누어 개발, 배포 및 확장이 용이하도록 하는 소프트웨어 설계 방법론입니다. 이번 포스팅에서는 "마이크로서비스 아키텍처", "설계", "구현"을 중심으로 마이크로서비스의 개념과 설계 원칙, 구현 방법을 상세히 설명하겠습니다. 예제를 통해 실습해 보겠습니다.


목차

  1. 마이크로서비스 아키텍처란?
  2. 마이크로서비스의 설계 원칙
  3. 마이크로서비스 구현 방법
    1. API 게이트웨이
    2. 서비스 디스커버리
    3. 데이터베이스 설계
  4. 예제 프로젝트: 간단한 전자상거래 시스템 구현
    1. 사용자 서비스
    2. 제품 서비스
    3. 주문 서비스
  5. 고급 주제
    1. 서비스 간 통신
    2. 보안
    3. 모니터링 및 로깅
  6. 결론
  7. 추천 태그

1. 마이크로서비스 아키텍처란?


마이크로서비스 아키텍처는 애플리케이션을 독립적으로 배포되고 확장 가능한 작은 서비스 단위로 나누는 설계 방법입니다. 각 서비스는 특정 도메인에 집중하며, 독립적인 데이터 저장소를 가질 수 있습니다. 이 접근 방식은 대규모 애플리케이션의 유지보수성과 확장성을 크게 향상시킵니다.


마이크로서비스의 장점


  1. 독립적 배포: 각 서비스는 독립적으로 배포 가능하여, 전체 시스템에 영향을 주지 않고도 업데이트할 수 있습니다.
  2. 확장성: 특정 서비스에 대한 수요가 증가할 때, 해당 서비스만 확장할 수 있습니다.
  3. 유지보수성: 작은 코드베이스는 이해하기 쉽고, 오류 수정이 용이합니다.
  4. 기술 독립성: 각 서비스는 다양한 기술 스택을 사용할 수 있습니다.

마이크로서비스의 단점


  1. 복잡성 증가: 서비스 간 통신, 데이터 일관성 관리 등의 복잡성이 증가합니다.
  2. 배포 및 운영 관리: 여러 서비스의 배포 및 운영을 관리하는 것이 복잡합니다.
  3. 네트워크 비용: 서비스 간 통신이 네트워크를 통해 이루어지므로 지연 시간이 발생할 수 있습니다.

2. 마이크로서비스의 설계 원칙


마이크로서비스 아키텍처를 성공적으로 설계하려면 몇 가지 주요 원칙을 따라야 합니다.


단일 책임 원칙


각 서비스는 하나의 책임을 가져야 하며, 이를 통해 서비스의 응집도를 높이고 결합도를 낮출 수 있습니다.


독립적 배포


각 서비스는 독립적으로 배포되고, 다른 서비스에 영향을 주지 않아야 합니다.


데이터베이스 분리


각 서비스는 독립적인 데이터 저장소를 가져야 합니다. 이를 통해 데이터 스키마 변경이 다른 서비스에 영향을 미치지 않게 됩니다.


서비스 디스커버리


서비스 간 통신을 원활하게 하기 위해 서비스 디스커버리 메커니즘을 사용하여 동적으로 서비스 위치를 찾아야 합니다.


3. 마이크로서비스 구현 방법


마이크로서비스 아키텍처를 구현하기 위해 몇 가지 핵심 컴포넌트를 설정해야 합니다.


API 게이트웨이


API 게이트웨이는 클라이언트 요청을 여러 마이크로서비스로 라우팅하는 진입점 역할을 합니다. 주요 기능은 다음과 같습니다:


  • 라우팅: 요청을 적절한 서비스로 전달
  • 인증 및 인가: 요청의 인증 및 인가 처리
  • 데이터 집계: 여러 서비스에서 데이터를 가져와 하나의 응답으로 결합

from flask import Flask, request, jsonify
import requests

app = Flask(__name__)

@app.route('/api/users', methods=['GET'])
def get_users():
    response = requests.get('http://user-service/api/users')
    return jsonify(response.json())

@app.route('/api/products', methods=['GET'])
def get_products():
    response = requests.get('http://product-service/api/products')
    return jsonify(response.json())

if __name__ == '__main__':
    app.run(port=8000)

서비스 디스커버리


서비스 디스커버리는 마이크로서비스의 동적 위치를 관리하는 메커니즘입니다. Consul, Eureka와 같은 도구를 사용하여 서비스 등록 및 검색을 관리할 수 있습니다.


from flask import Flask, jsonify
import requests

app = Flask(__name__)

@app.route('/api/hello', methods=['GET'])
def hello():
    # 서비스 디스커버리를 사용하여 다른 서비스의 위치를 찾음
    service_url = discover_service('hello-service')
    response = requests.get(service_url)
    return jsonify(response.json())

def discover_service(service_name):
    # Consul 또는 Eureka를 사용하여 서비스 위치를 조회
    response = requests.get(f'http://discovery-service/v1/catalog/service/{service_name}')
    service_info = response.json()[0]
    return f"http://{service_info['ServiceAddress']}:{service_info['ServicePort']}"

if __name__ == '__main__':
    app.run(port=8001)

데이터베이스 설계


각 마이크로서비스는 독립적인 데이터베이스를 가져야 합니다. 이를 통해 데이터베이스 스키마 변경이 다른 서비스에 영향을 미치지 않습니다.


-- 사용자 서비스 데이터베이스 스키마
CREATE TABLE users (
    user_id SERIAL PRIMARY KEY,
    username VARCHAR(255) NOT NULL,
    email VARCHAR(255) UNIQUE NOT NULL
);

-- 제품 서비스 데이터베이스 스키마
CREATE TABLE products (
    product_id SERIAL PRIMARY KEY,
    name VARCHAR(255) NOT NULL,
    price DECIMAL NOT NULL
);

4. 예제 프로젝트: 간단한 전자상거래 시스템 구현


간단한 전자상거래 시스템을 마이크로서비스 아키텍처로 구현해 보겠습니다. 주요 서비스는 사용자 서비스, 제품 서비스, 주문 서비스로 구성됩니다.


1. 사용자 서비스


사용자 정보를 관리하는 서비스입니다.


from flask import Flask, request, jsonify
from flask_sqlalchemy import SQLAlchemy

app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'postgresql://user:password@localhost/userdb'
db = SQLAlchemy(app)

class User(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    username = db.Column(db.String(80), unique=True, nullable=False)
    email = db.Column(db.String(120), unique=True, nullable=False)

@app.route('/api/users', methods=['POST'])
def create_user():
    data = request.get_json()
    new_user = User(username=data['username'], email=data['email'])
    db.session.add(new_user)
    db.session.commit()
    return jsonify({'message': 'User created successfully'})

@app.route('/api/users', methods=['GET'])
def get_users():
    users = User.query.all()
    return jsonify([{'id': user.id, 'username': user.username, 'email': user.email} for user in users])

if __name__ == '__main__':
    app.run(port=5001)

2. 제품 서비스


제품 정보를 관리하는 서비스입니다.


from flask import Flask, request, jsonify
from flask_sqlalchemy import SQLAlchemy

app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'postgresql://user:password@localhost/productdb'
db = SQLAlchemy(app)

class Product(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String(80), unique=True, nullable=False)
    price = db.Column(db.Float, nullable=False)

@app.route('/api/products', methods=['POST'])
def create_product():
    data = request.get_json()
    new_product = Product(name=data['name'], price=data['price'])
    db.session.add(new_product)
    db.session.commit()
    return jsonify({'message': 'Product created successfully'})

@app.route('/api/products', methods=['GET'])
def get_products():
    products = Product.query.all()
    return jsonify([{'id': product.id, 'name': product.name, 'price': product.price} for product in products])

if __name__ == '__main__':
    app.run(port=5002)

3. 주문 서비스


주문 정보를 관리하는 서비스입니다.


from flask import Flask, request, jsonify
from flask_sqlalchemy import SQLAlchemy

app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'postgresql://user:password@localhost/orderdb'
db = SQLAlchemy(app)

class Order(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    user_id = db.Column(db.Integer, nullable=False)
    product_id = db.Column(db.Integer, nullable=False)
    quantity = db.Column(db.Integer, nullable=False)

@app.route('/api/orders', methods=['POST'])
def create_order():
    data = request.get_json()
    new_order = Order(user_id=data['user_id'], product_id=data['product_id'], quantity=data['quantity'])
    db.session.add(new_order)
    db.session.commit()
    return jsonify({'message': 'Order created successfully'})

@app.route('/api/orders', methods=['GET'])
def get_orders():
    orders = Order.query.all()
    return jsonify([{'id': order.id, 'user_id': order.user_id, 'product_id': order.product_id, 'quantity': order.quantity} for order in orders])

if __name__ == '__main__':
    app.run(port=5003)

5. 고급 주제


1. 서비스 간 통신


마이크로서비스 간 통신은 HTTP 요청, 메시지 큐 등을 사용하여 구현할 수 있습니다. 메시지 큐를 사용하면 서비스 간 비동기 통신을 구현할 수 있습니다.


import pika

def send_message(queue_name, message):
    connection = pika.BlockingConnection(pika.ConnectionParameters('localhost'))
    channel = connection.channel()
    channel.queue_declare(queue=queue_name)
    channel.basic_publish(exchange='', routing_key=queue_name, body=message)
    connection.close()

send_message('order_queue', 'New order placed')

2. 보안


각 서비스는 인증 및 인가를 통해 보안을 강화해야 합니다. JWT(JSON Web Token)를 사용하여 사용자 인증을 구현할 수 있습니다.


from flask import Flask, request, jsonify
import jwt
import datetime

app = Flask(__name__)
app.config['SECRET_KEY'] = 'your_secret_key'

@app.route('/api/login', methods=['POST'])
def login():
    auth = request.authorization
    if auth and auth.password == 'password':
        token = jwt.encode({'user': auth.username, 'exp': datetime.datetime.utcnow() + datetime.timedelta(minutes=30)}, app.config['SECRET_KEY'])
        return jsonify({'token': token})
    return jsonify({'message': 'Could not verify'}), 401

@app.route('/api/protected', methods=['GET'])
def protected():
    token = request.headers.get('x-access-tokens')
    if not token:
        return jsonify({'message': 'Token is missing'}), 401
    try:
        data = jwt.decode(token, app.config['SECRET_KEY'], algorithms=["HS256"])
    except:
        return jsonify({'message': 'Token is invalid'}), 401
    return jsonify({'message': 'Protected route'})

if __name__ == '__main__':
    app.run(port=5000)

3. 모니터링 및 로깅


각 서비스의 상태를 모니터링하고 로그를 수집하여 시스템의 건강 상태를 유지할 수 있습니다. Prometheus와 Grafana를 사용하여 모니터링 대시보드를 구축할 수 있습니다.


from flask import Flask, request
import logging

app = Flask(__name__)

logging.basicConfig(level=logging.INFO)

@app.route('/api/some_endpoint', methods=['GET'])
def some_endpoint():
    app.logger.info('Endpoint was called')
    return 'Hello, World!'

if __name__ == '__main__':
    app.run(port=5000)

6. 결론


마이크로서비스 아키텍처는 대규모 애플리케이션을 보다 유연하고 확장 가능하게 만들어 줍니다. 이번 포스팅에서는 마이크로서비스의 개념과 설계 원칙, 구현 방법을 설명하고, 간단한 예제 프로젝트를 통해 실습해 보았습니다. 고급 주제로 서비스 간 통신, 보안, 모니터링 및 로깅을 다루어 마이크로서비스 아키텍처의 중요 요소를 설명하였습니다. 이러한 기술을 잘 활용하면 더 나은 확장성과 유지보수성을 갖춘 애플리케이션을 개발할 수 있습니다.

이 포스팅이 마이크로서비스 아키텍처를 이해하고 구현하는 데 도움이 되길 바랍니다. 질문이나 추가 정보가 필요하시면 언제든지 댓글로 남겨주세요.

다음 이전