Generate Vapor project.
This commit is contained in:
2
.dockerignore
Normal file
2
.dockerignore
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
.build/
|
||||||
|
.swiftpm/
|
11
.gitignore
vendored
Normal file
11
.gitignore
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
Packages
|
||||||
|
.build
|
||||||
|
xcuserdata
|
||||||
|
*.xcodeproj
|
||||||
|
DerivedData/
|
||||||
|
.DS_Store
|
||||||
|
db.sqlite
|
||||||
|
.swiftpm
|
||||||
|
.env
|
||||||
|
.env.*
|
||||||
|
!.env.example
|
3
.vscode/extensions.json
vendored
Normal file
3
.vscode/extensions.json
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"recommendations": ["swiftlang.swift-vscode", "Vapor.vapor-vscode"]
|
||||||
|
}
|
89
Dockerfile
Normal file
89
Dockerfile
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
# ================================
|
||||||
|
# Build image
|
||||||
|
# ================================
|
||||||
|
FROM swift:6.0-noble AS build
|
||||||
|
|
||||||
|
# Install OS updates
|
||||||
|
RUN export DEBIAN_FRONTEND=noninteractive DEBCONF_NONINTERACTIVE_SEEN=true \
|
||||||
|
&& apt-get -q update \
|
||||||
|
&& apt-get -q dist-upgrade -y \
|
||||||
|
&& apt-get install -y libjemalloc-dev
|
||||||
|
|
||||||
|
# Set up a build area
|
||||||
|
WORKDIR /build
|
||||||
|
|
||||||
|
# First just resolve dependencies.
|
||||||
|
# This creates a cached layer that can be reused
|
||||||
|
# as long as your Package.swift/Package.resolved
|
||||||
|
# files do not change.
|
||||||
|
COPY ./Package.* ./
|
||||||
|
RUN swift package resolve \
|
||||||
|
$([ -f ./Package.resolved ] && echo "--force-resolved-versions" || true)
|
||||||
|
|
||||||
|
# Copy entire repo into container
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
# Build the application, with optimizations, with static linking, and using jemalloc
|
||||||
|
# N.B.: The static version of jemalloc is incompatible with the static Swift runtime.
|
||||||
|
RUN swift build -c release \
|
||||||
|
--product stella \
|
||||||
|
--static-swift-stdlib \
|
||||||
|
-Xlinker -ljemalloc
|
||||||
|
|
||||||
|
# Switch to the staging area
|
||||||
|
WORKDIR /staging
|
||||||
|
|
||||||
|
# Copy main executable to staging area
|
||||||
|
RUN cp "$(swift build --package-path /build -c release --show-bin-path)/stella" ./
|
||||||
|
|
||||||
|
# Copy static swift backtracer binary to staging area
|
||||||
|
RUN cp "/usr/libexec/swift/linux/swift-backtrace-static" ./
|
||||||
|
|
||||||
|
# Copy resources bundled by SPM to staging area
|
||||||
|
RUN find -L "$(swift build --package-path /build -c release --show-bin-path)/" -regex '.*\.resources$' -exec cp -Ra {} ./ \;
|
||||||
|
|
||||||
|
# Copy any resources from the public directory and views directory if the directories exist
|
||||||
|
# Ensure that by default, neither the directory nor any of its contents are writable.
|
||||||
|
RUN [ -d /build/Public ] && { mv /build/Public ./Public && chmod -R a-w ./Public; } || true
|
||||||
|
RUN [ -d /build/Resources ] && { mv /build/Resources ./Resources && chmod -R a-w ./Resources; } || true
|
||||||
|
|
||||||
|
# ================================
|
||||||
|
# Run image
|
||||||
|
# ================================
|
||||||
|
FROM ubuntu:noble
|
||||||
|
|
||||||
|
# Make sure all system packages are up to date, and install only essential packages.
|
||||||
|
RUN export DEBIAN_FRONTEND=noninteractive DEBCONF_NONINTERACTIVE_SEEN=true \
|
||||||
|
&& apt-get -q update \
|
||||||
|
&& apt-get -q dist-upgrade -y \
|
||||||
|
&& apt-get -q install -y \
|
||||||
|
libjemalloc2 \
|
||||||
|
ca-certificates \
|
||||||
|
tzdata \
|
||||||
|
# If your app or its dependencies import FoundationNetworking, also install `libcurl4`.
|
||||||
|
# libcurl4 \
|
||||||
|
# If your app or its dependencies import FoundationXML, also install `libxml2`.
|
||||||
|
# libxml2 \
|
||||||
|
&& rm -r /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# Create a vapor user and group with /app as its home directory
|
||||||
|
RUN useradd --user-group --create-home --system --skel /dev/null --home-dir /app vapor
|
||||||
|
|
||||||
|
# Switch to the new home directory
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
# Copy built executable and any staged resources from builder
|
||||||
|
COPY --from=build --chown=vapor:vapor /staging /app
|
||||||
|
|
||||||
|
# Provide configuration needed by the built-in crash reporter and some sensible default behaviors.
|
||||||
|
ENV SWIFT_BACKTRACE=enable=yes,sanitize=yes,threads=all,images=all,interactive=no,swift-backtrace=./swift-backtrace-static
|
||||||
|
|
||||||
|
# Ensure all further commands run as the vapor user
|
||||||
|
USER vapor:vapor
|
||||||
|
|
||||||
|
# Let Docker bind to port 8080
|
||||||
|
EXPOSE 8080
|
||||||
|
|
||||||
|
# Start the Vapor service when the image is run, default to listening on 8080 in production environment
|
||||||
|
ENTRYPOINT ["./stella"]
|
||||||
|
CMD ["serve", "--env", "production", "--hostname", "0.0.0.0", "--port", "8080"]
|
44
Package.swift
Normal file
44
Package.swift
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
// swift-tools-version:6.0
|
||||||
|
import PackageDescription
|
||||||
|
|
||||||
|
let package = Package(
|
||||||
|
name: "stella",
|
||||||
|
platforms: [
|
||||||
|
.macOS(.v13)
|
||||||
|
],
|
||||||
|
dependencies: [
|
||||||
|
// 💧 A server-side Swift web framework.
|
||||||
|
.package(url: "https://github.com/vapor/vapor.git", from: "4.110.1"),
|
||||||
|
// 🗄 An ORM for SQL and NoSQL databases.
|
||||||
|
.package(url: "https://github.com/vapor/fluent.git", from: "4.9.0"),
|
||||||
|
// 🪶 Fluent driver for SQLite.
|
||||||
|
.package(url: "https://github.com/vapor/fluent-sqlite-driver.git", from: "4.6.0"),
|
||||||
|
// 🔵 Non-blocking, event-driven networking for Swift. Used for custom executors
|
||||||
|
.package(url: "https://github.com/apple/swift-nio.git", from: "2.65.0"),
|
||||||
|
],
|
||||||
|
targets: [
|
||||||
|
.executableTarget(
|
||||||
|
name: "stella",
|
||||||
|
dependencies: [
|
||||||
|
.product(name: "Fluent", package: "fluent"),
|
||||||
|
.product(name: "FluentSQLiteDriver", package: "fluent-sqlite-driver"),
|
||||||
|
.product(name: "Vapor", package: "vapor"),
|
||||||
|
.product(name: "NIOCore", package: "swift-nio"),
|
||||||
|
.product(name: "NIOPosix", package: "swift-nio"),
|
||||||
|
],
|
||||||
|
swiftSettings: swiftSettings
|
||||||
|
),
|
||||||
|
.testTarget(
|
||||||
|
name: "stellaTests",
|
||||||
|
dependencies: [
|
||||||
|
.target(name: "stella"),
|
||||||
|
.product(name: "VaporTesting", package: "vapor"),
|
||||||
|
],
|
||||||
|
swiftSettings: swiftSettings
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
var swiftSettings: [SwiftSetting] { [
|
||||||
|
.enableUpcomingFeature("ExistentialAny"),
|
||||||
|
] }
|
0
Public/.gitkeep
Normal file
0
Public/.gitkeep
Normal file
27
README.md
Normal file
27
README.md
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
# stella
|
||||||
|
|
||||||
|
💧 A project built with the Vapor web framework.
|
||||||
|
|
||||||
|
## Getting Started
|
||||||
|
|
||||||
|
To build the project using the Swift Package Manager, run the following command in the terminal from the root of the project:
|
||||||
|
```bash
|
||||||
|
swift build
|
||||||
|
```
|
||||||
|
|
||||||
|
To run the project and start the server, use the following command:
|
||||||
|
```bash
|
||||||
|
swift run
|
||||||
|
```
|
||||||
|
|
||||||
|
To execute tests, use the following command:
|
||||||
|
```bash
|
||||||
|
swift test
|
||||||
|
```
|
||||||
|
|
||||||
|
### See more
|
||||||
|
|
||||||
|
- [Vapor Website](https://vapor.codes)
|
||||||
|
- [Vapor Documentation](https://docs.vapor.codes)
|
||||||
|
- [Vapor GitHub](https://github.com/vapor)
|
||||||
|
- [Vapor Community](https://github.com/vapor-community)
|
0
Sources/stella/Controllers/.gitkeep
Normal file
0
Sources/stella/Controllers/.gitkeep
Normal file
37
Sources/stella/Controllers/TodoController.swift
Normal file
37
Sources/stella/Controllers/TodoController.swift
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
import Fluent
|
||||||
|
import Vapor
|
||||||
|
|
||||||
|
struct TodoController: RouteCollection {
|
||||||
|
func boot(routes: any RoutesBuilder) throws {
|
||||||
|
let todos = routes.grouped("todos")
|
||||||
|
|
||||||
|
todos.get(use: self.index)
|
||||||
|
todos.post(use: self.create)
|
||||||
|
todos.group(":todoID") { todo in
|
||||||
|
todo.delete(use: self.delete)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Sendable
|
||||||
|
func index(req: Request) async throws -> [TodoDTO] {
|
||||||
|
try await Todo.query(on: req.db).all().map { $0.toDTO() }
|
||||||
|
}
|
||||||
|
|
||||||
|
@Sendable
|
||||||
|
func create(req: Request) async throws -> TodoDTO {
|
||||||
|
let todo = try req.content.decode(TodoDTO.self).toModel()
|
||||||
|
|
||||||
|
try await todo.save(on: req.db)
|
||||||
|
return todo.toDTO()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Sendable
|
||||||
|
func delete(req: Request) async throws -> HTTPStatus {
|
||||||
|
guard let todo = try await Todo.find(req.parameters.get("todoID"), on: req.db) else {
|
||||||
|
throw Abort(.notFound)
|
||||||
|
}
|
||||||
|
|
||||||
|
try await todo.delete(on: req.db)
|
||||||
|
return .noContent
|
||||||
|
}
|
||||||
|
}
|
17
Sources/stella/DTOs/TodoDTO.swift
Normal file
17
Sources/stella/DTOs/TodoDTO.swift
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import Fluent
|
||||||
|
import Vapor
|
||||||
|
|
||||||
|
struct TodoDTO: Content {
|
||||||
|
var id: UUID?
|
||||||
|
var title: String?
|
||||||
|
|
||||||
|
func toModel() -> Todo {
|
||||||
|
let model = Todo()
|
||||||
|
|
||||||
|
model.id = self.id
|
||||||
|
if let title = self.title {
|
||||||
|
model.title = title
|
||||||
|
}
|
||||||
|
return model
|
||||||
|
}
|
||||||
|
}
|
14
Sources/stella/Migrations/CreateTodo.swift
Normal file
14
Sources/stella/Migrations/CreateTodo.swift
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import Fluent
|
||||||
|
|
||||||
|
struct CreateTodo: AsyncMigration {
|
||||||
|
func prepare(on database: any Database) async throws {
|
||||||
|
try await database.schema("todos")
|
||||||
|
.id()
|
||||||
|
.field("title", .string, .required)
|
||||||
|
.create()
|
||||||
|
}
|
||||||
|
|
||||||
|
func revert(on database: any Database) async throws {
|
||||||
|
try await database.schema("todos").delete()
|
||||||
|
}
|
||||||
|
}
|
29
Sources/stella/Models/Todo.swift
Normal file
29
Sources/stella/Models/Todo.swift
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import Fluent
|
||||||
|
import struct Foundation.UUID
|
||||||
|
|
||||||
|
/// Property wrappers interact poorly with `Sendable` checking, causing a warning for the `@ID` property
|
||||||
|
/// It is recommended you write your model with sendability checking on and then suppress the warning
|
||||||
|
/// afterwards with `@unchecked Sendable`.
|
||||||
|
final class Todo: Model, @unchecked Sendable {
|
||||||
|
static let schema = "todos"
|
||||||
|
|
||||||
|
@ID(key: .id)
|
||||||
|
var id: UUID?
|
||||||
|
|
||||||
|
@Field(key: "title")
|
||||||
|
var title: String
|
||||||
|
|
||||||
|
init() { }
|
||||||
|
|
||||||
|
init(id: UUID? = nil, title: String) {
|
||||||
|
self.id = id
|
||||||
|
self.title = title
|
||||||
|
}
|
||||||
|
|
||||||
|
func toDTO() -> TodoDTO {
|
||||||
|
.init(
|
||||||
|
id: self.id,
|
||||||
|
title: self.$title.value
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
17
Sources/stella/configure.swift
Normal file
17
Sources/stella/configure.swift
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import NIOSSL
|
||||||
|
import Fluent
|
||||||
|
import FluentSQLiteDriver
|
||||||
|
import Vapor
|
||||||
|
|
||||||
|
// configures your application
|
||||||
|
public func configure(_ app: Application) async throws {
|
||||||
|
// uncomment to serve files from /Public folder
|
||||||
|
// app.middleware.use(FileMiddleware(publicDirectory: app.directory.publicDirectory))
|
||||||
|
|
||||||
|
app.databases.use(DatabaseConfigurationFactory.sqlite(.file("db.sqlite")), as: .sqlite)
|
||||||
|
|
||||||
|
app.migrations.add(CreateTodo())
|
||||||
|
|
||||||
|
// register routes
|
||||||
|
try routes(app)
|
||||||
|
}
|
31
Sources/stella/entrypoint.swift
Normal file
31
Sources/stella/entrypoint.swift
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import Vapor
|
||||||
|
import Logging
|
||||||
|
import NIOCore
|
||||||
|
import NIOPosix
|
||||||
|
|
||||||
|
@main
|
||||||
|
enum Entrypoint {
|
||||||
|
static func main() async throws {
|
||||||
|
var env = try Environment.detect()
|
||||||
|
try LoggingSystem.bootstrap(from: &env)
|
||||||
|
|
||||||
|
let app = try await Application.make(env)
|
||||||
|
|
||||||
|
// This attempts to install NIO as the Swift Concurrency global executor.
|
||||||
|
// You can enable it if you'd like to reduce the amount of context switching between NIO and Swift Concurrency.
|
||||||
|
// Note: this has caused issues with some libraries that use `.wait()` and cleanly shutting down.
|
||||||
|
// If enabled, you should be careful about calling async functions before this point as it can cause assertion failures.
|
||||||
|
// let executorTakeoverSuccess = NIOSingletons.unsafeTryInstallSingletonPosixEventLoopGroupAsConcurrencyGlobalExecutor()
|
||||||
|
// app.logger.debug("Tried to install SwiftNIO's EventLoopGroup as Swift's global concurrency executor", metadata: ["success": .stringConvertible(executorTakeoverSuccess)])
|
||||||
|
|
||||||
|
do {
|
||||||
|
try await configure(app)
|
||||||
|
try await app.execute()
|
||||||
|
} catch {
|
||||||
|
app.logger.report(error: error)
|
||||||
|
try? await app.asyncShutdown()
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
try await app.asyncShutdown()
|
||||||
|
}
|
||||||
|
}
|
14
Sources/stella/routes.swift
Normal file
14
Sources/stella/routes.swift
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import Fluent
|
||||||
|
import Vapor
|
||||||
|
|
||||||
|
func routes(_ app: Application) throws {
|
||||||
|
app.get { req async in
|
||||||
|
"It works!"
|
||||||
|
}
|
||||||
|
|
||||||
|
app.get("hello") { req async -> String in
|
||||||
|
"Hello, world!"
|
||||||
|
}
|
||||||
|
|
||||||
|
try app.register(collection: TodoController())
|
||||||
|
}
|
81
Tests/stellaTests/stellaTests.swift
Normal file
81
Tests/stellaTests/stellaTests.swift
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
@testable import stella
|
||||||
|
import VaporTesting
|
||||||
|
import Testing
|
||||||
|
import Fluent
|
||||||
|
|
||||||
|
@Suite("App Tests with DB", .serialized)
|
||||||
|
struct stellaTests {
|
||||||
|
private func withApp(_ test: (Application) async throws -> ()) async throws {
|
||||||
|
let app = try await Application.make(.testing)
|
||||||
|
do {
|
||||||
|
try await configure(app)
|
||||||
|
try await app.autoMigrate()
|
||||||
|
try await test(app)
|
||||||
|
try await app.autoRevert()
|
||||||
|
} catch {
|
||||||
|
try? await app.autoRevert()
|
||||||
|
try await app.asyncShutdown()
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
try await app.asyncShutdown()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test("Test Hello World Route")
|
||||||
|
func helloWorld() async throws {
|
||||||
|
try await withApp { app in
|
||||||
|
try await app.testing().test(.GET, "hello", afterResponse: { res async in
|
||||||
|
#expect(res.status == .ok)
|
||||||
|
#expect(res.body.string == "Hello, world!")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test("Getting all the Todos")
|
||||||
|
func getAllTodos() async throws {
|
||||||
|
try await withApp { app in
|
||||||
|
let sampleTodos = [Todo(title: "sample1"), Todo(title: "sample2")]
|
||||||
|
try await sampleTodos.create(on: app.db)
|
||||||
|
|
||||||
|
try await app.testing().test(.GET, "todos", afterResponse: { res async throws in
|
||||||
|
#expect(res.status == .ok)
|
||||||
|
#expect(try res.content.decode([TodoDTO].self) == sampleTodos.map { $0.toDTO()} )
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test("Creating a Todo")
|
||||||
|
func createTodo() async throws {
|
||||||
|
let newDTO = TodoDTO(id: nil, title: "test")
|
||||||
|
|
||||||
|
try await withApp { app in
|
||||||
|
try await app.testing().test(.POST, "todos", beforeRequest: { req in
|
||||||
|
try req.content.encode(newDTO)
|
||||||
|
}, afterResponse: { res async throws in
|
||||||
|
#expect(res.status == .ok)
|
||||||
|
let models = try await Todo.query(on: app.db).all()
|
||||||
|
#expect(models.map({ $0.toDTO().title }) == [newDTO.title])
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test("Deleting a Todo")
|
||||||
|
func deleteTodo() async throws {
|
||||||
|
let testTodos = [Todo(title: "test1"), Todo(title: "test2")]
|
||||||
|
|
||||||
|
try await withApp { app in
|
||||||
|
try await testTodos.create(on: app.db)
|
||||||
|
|
||||||
|
try await app.testing().test(.DELETE, "todos/\(testTodos[0].requireID())", afterResponse: { res async throws in
|
||||||
|
#expect(res.status == .noContent)
|
||||||
|
let model = try await Todo.find(testTodos[0].id, on: app.db)
|
||||||
|
#expect(model == nil)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extension TodoDTO: Equatable {
|
||||||
|
public static func == (lhs: Self, rhs: Self) -> Bool {
|
||||||
|
lhs.id == rhs.id && lhs.title == rhs.title
|
||||||
|
}
|
||||||
|
}
|
47
docker-compose.yml
Normal file
47
docker-compose.yml
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
# Docker Compose file for Vapor
|
||||||
|
#
|
||||||
|
# Install Docker on your system to run and test
|
||||||
|
# your Vapor app in a production-like environment.
|
||||||
|
#
|
||||||
|
# Note: This file is intended for testing and does not
|
||||||
|
# implement best practices for a production deployment.
|
||||||
|
#
|
||||||
|
# Learn more: https://docs.docker.com/compose/reference/
|
||||||
|
#
|
||||||
|
# Build images: docker compose build
|
||||||
|
# Start app: docker compose up app
|
||||||
|
# Stop all: docker compose down
|
||||||
|
#
|
||||||
|
|
||||||
|
x-shared_environment: &shared_environment
|
||||||
|
LOG_LEVEL: ${LOG_LEVEL:-debug}
|
||||||
|
|
||||||
|
services:
|
||||||
|
app:
|
||||||
|
image: stella:latest
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
environment:
|
||||||
|
<<: *shared_environment
|
||||||
|
ports:
|
||||||
|
- '8080:8080'
|
||||||
|
# user: '0' # uncomment to run as root for testing purposes even though Dockerfile defines 'vapor' user.
|
||||||
|
command: ["serve", "--env", "production", "--hostname", "0.0.0.0", "--port", "8080"]
|
||||||
|
migrate:
|
||||||
|
image: stella:latest
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
environment:
|
||||||
|
<<: *shared_environment
|
||||||
|
command: ["migrate", "--yes"]
|
||||||
|
deploy:
|
||||||
|
replicas: 0
|
||||||
|
revert:
|
||||||
|
image: stella:latest
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
environment:
|
||||||
|
<<: *shared_environment
|
||||||
|
command: ["migrate", "--revert", "--yes"]
|
||||||
|
deploy:
|
||||||
|
replicas: 0
|
Reference in New Issue
Block a user