Skip to content

apache-edge/csv

Repository files navigation

CSV

A Cross-Platform Swift 6 Library for CSV Parsing with Concurrency Support

Swift Platforms macOS Linux Windows License

A cross-platform Swift 6 library for CSV parsing with concurrency support. Works on Apple platforms (iOS, macOS, tvOS, watchOS), Linux, Android, and Windows.

Features

  • ✅ Cross-platform: Works on all Swift-supported platforms
  • ✅ Swift Concurrency: Built with Swift's concurrency model
  • ✅ Modern Swift: Designed for Swift 6
  • ✅ Codable Support: Simple conversion between CSV and Swift types
  • ✅ Flexible: Works with different delimiters, quote characters, and header options
  • ✅ Type-safe: Strong typing for all operations
  • ✅ Performance: Parallel processing of CSV rows

Installation

Swift Package Manager

Add the following to your Package.swift file:

dependencies: [
    .package(url: "https://github.com/apache-edge/csv.git", from: "0.1.0")
]

Basic Usage

Parse a CSV string

import CSV

let csvString = """
name,age,email
John,30,[email protected]
Alice,25,[email protected]
Bob,40,[email protected]
"""

do {
    let csv = try CSV(content: csvString)
    
    // Access headers
    print(csv.headers) // ["name", "age", "email"]
    
    // Access rows as dictionaries
    for row in csv.namedRows {
        print("\(row["name"]!) is \(row["age"]!) years old")
    }
    
    // Access a specific column
    if let emails = csv.column(named: "email") {
        print("All emails: \(emails)")
    }
} catch {
    print("Error parsing CSV: \(error)")
}

Parse a CSV file

import CSV

do {
    // Synchronous API
    let csv = try CSV(path: "path/to/file.csv")
    
    // Asynchronous API
    let csvAsync = try await CSV.load(path: "path/to/file.csv")
    
    // Work with data...
} catch {
    print("Error parsing CSV: \(error)")
}

Concurrency Features

Process rows in parallel using Swift's concurrency model:

import CSV

let csv = try CSV(content: csvString)

// Process all rows concurrently and collect results
let results = try await csv.processRowsConcurrently { index, row in
    // Do some heavy processing...
    return processedValue
}

// Process named rows (with headers)
let namedResults = try await csv.processNamedRowsConcurrently { index, row in
    return row["name"]!.uppercased()
}

Codable Support

Easily convert between CSV and your Swift types:

import CSV

// Define a model conforming to CSVCodable
struct Person: CSVCodable {
    let name: String
    let age: Int
    let email: String
    
    // Required initializer
    init(csvRow row: [String: String]) throws {
        self.name = row["name"] ?? ""
        self.age = Int(row["age"] ?? "0") ?? 0
        self.email = row["email"] ?? ""
    }
}

// Decode from CSV
let csv = try CSV(content: csvString)
let people = try csv.decode(Person.self)

// Work with strongly-typed objects
for person in people {
    print("\(person.name) is \(person.age) years old")
}

// Decode concurrently for better performance
let peopleConcurrent = try await csv.decodeConcurrently(Person.self)

// Encode back to CSV
let newCsvString = try CSV.encode(people)

Cell Parsing

Parse individual cells with type safety and error handling:

import CSV

let csv = try CSV(content: csvString)

// Parse cells with type inference
let name: String = try csv.parse(at: 0, column: "name")
let age: Int = try csv.parse(at: 0, column: "age")
let score: Double = try csv.parse(at: 0, column: "score")
let isActive: Bool = try csv.parse(at: 0, column: "active")

// Parse cells by column index
let firstName: String = try csv.parse(at: 0, columnIndex: 0)

// Parse cells with custom logic
let formattedName = try csv.parseCell(at: 0, column: "name") { value in
    return value.uppercased()
}

// Parse dates with custom formatter
let dateFormatter = DateFormatter()
dateFormatter.dateFormat = "yyyy-MM-dd"
let date = try csv.parseCell(at: 0, column: "date") { value in
    guard let date = dateFormatter.date(from: value) else {
        throw CSVCellParsingError.invalidValue(value, targetType: "Date")
    }
    return date
}

// Parse cells asynchronously
let result = try await csv.parseCellAsync(at: 0, column: "data") { value in
    // Perform async work like API calls or complex processing
    return try await processValueAsynchronously(value)
}

Customization

Configure the CSV parser to match your data format:

// Custom delimiter (semicolon)
let csv = try CSV(content: csvContent, delimiter: ";")

// Tab-delimited files
let tsvFile = try CSV(content: tsvContent, delimiter: "\t")

// Single quote for quoted fields
let csv = try CSV(content: csvContent, quoteCharacter: "'")

// CSV without headers
let csv = try CSV(content: csvContent, hasHeaders: false)

License

CSV is available under the MIT license.

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Packages

No packages published

Languages