Official Python client for Apploi Partner API with high-level abstractions
Project description
Apploi Partner API - Python Client
The official Python client library for the Apploi Partner API. This library provides a clean, type-safe interface for integrating with Apploi's applicant tracking system, allowing you to programmatically access and manage job applications, candidates, and related data.
What is the Apploi Partner API?
The Apploi Partner API enables seamless integration with Apploi's platform, giving you programmatic access to:
- Job Applications: Retrieve, filter, and process candidate applications across your organization
- Candidate Data: Access detailed candidate information including contact details, application history, and status
- Application Workflow: Track application status changes, state transitions, and timeline data
- Team Management: Filter and organize data by teams and their associated jobs
Why Use This Client Library?
While you could interact with the API directly using HTTP requests, this Python client provides significant advantages:
🛡️ Type Safety & Validation
- Strong typing with full IDE autocomplete support
- Automatic parameter validation before API calls
- Structured data models with proper Python types (
int,datetime,Enum)
🚀 Developer Experience
- Native Python idioms - use
date(2024, 1, 1)instead of"2024-01-01" - Intuitive method names and parameter handling
- Comprehensive error handling with descriptive messages
- Built-in pagination support
📦 Production Ready
- Automatic retry logic for transient failures
- Proper timeout handling and connection management
- Extensive test coverage and validation
- Clear documentation and examples
Installation
pip install apploi-partner-api
Quick Start
from apploi_partner_api import ApploiClient
from datetime import date
# Initialize client
client = ApploiClient(api_key="your-api-key")
# Get job applications with native Python types
job_applications = client.get_job_applications(
job_id=12345,
updated_after=date(2024, 1, 1),
limit=50
)
# Work with strongly-typed results
for application in job_applications:
print(f"{application.candidate.name} - {application.status.value}")
print(f" Email: {application.email}")
print(f" Applied: {application.applied_at}")
print(f" Job: {application.job.name}")
Key Features
🎯 Simple Integration
- Single
pip install- no complex setup required - Works with Python 3.7+ across all major platforms
- Minimal dependencies for easy deployment
🔧 Native Python Experience
- Use native Python types:
int,date,datetime,bool - Automatic type conversion and validation
- Pythonic method names and parameter handling
- No need to manually format dates or convert integers to strings
🛡️ Robust Error Handling
- Custom exception hierarchy for different error types
- Descriptive error messages with actionable guidance
- Automatic retry logic for transient network issues
- Proper handling of rate limits and API quotas
📊 Rich Data Models
- Structured dataclasses with full type annotations
- Nested models for complex data (Job, Candidate, Application)
- Convenient properties for common operations
- Access to both processed and raw API response data
🚀 Production Ready
- Comprehensive test coverage and validation
- Built-in pagination for large datasets
- Configurable timeouts and connection settings
- Thread-safe for concurrent usage
API Reference
Client Initialization
from apploi_partner_api import ApploiClient
client = ApploiClient(
api_key="your-api-key", # Required
base_url=None, # Optional, defaults to production
timeout=60 # Optional, request timeout in seconds
)
get_job_applications()
Retrieve job applications with filtering and pagination.
job_applications = client.get_job_applications(
state="applied", # Filter by status
query="Smith", # Search query filters by name, phone number, or email
limit=50, # Max results (1-1000)
team_id=123, # Team ID (int or str)
updated_before=datetime.now(), # Before date
offset=0, # Pagination offset
job_id=456, # Job ID (int or str)
updated_after=date(2024, 1, 1) # After date
)
Parameters:
- All parameters are optional
- IDs accept
intorstr - Dates accept
date,datetime, or ISO string - Limit must be 1-1000
- Offset must be ≥ 0
Returns: JobApplicationsList with:
items: List ofJobApplicationobjectstotal: Total count availablelimit: Page sizeoffset: Current offsethas_more: Boolean for more pages
Domain Models
JobApplication
@dataclass
class JobApplication:
# Identifiers
id: int # Application ID
candidate_id: int # Candidate ID
job_id: int # Job ID
# Required fields
status: ApplicationStatus # Application status
job: Job # Job details (nested)
candidate: Candidate # Candidate details (nested)
email: str # Applicant email
applicant_state: str # Current state
applicant_state_id: int # State ID
# DateTime/State Management
date_current_state_assigned: Optional[datetime]
previous_applicant_state: Optional[str]
previous_applicant_state_id: Optional[int]
date_previous_state_assigned: Optional[datetime]
applicant_updated: Optional[datetime]
date_created: Optional[datetime]
application_source: Optional[str]
# Optional fields
first_name: Optional[str]
last_name: Optional[str]
phone: Optional[str]
applied_at: Optional[datetime]
updated_at: Optional[datetime]
team_id: Optional[int]
@property
def name(self) -> str:
"""Full name (first + last)"""
Job
@dataclass
class Job:
id: Optional[int]
name: Optional[str] # Job title
archived: Optional[bool] # Is archived
archived_date: Optional[datetime]
address: Optional[str] # Job location
source: Optional[str] # Job source
brand_name: Optional[str] # Company brand
job_code: Optional[str] # Internal job code
creator_id: Optional[int] # Creator ID
team_id: Optional[int] # Team ID
team_name: Optional[str] # Team name
role_type: Optional[str] # Role type
longitude: Optional[float] # Geo coordinates
latitude: Optional[float]
Candidate
@dataclass
class Candidate:
id: int # Candidate ID
first_name: Optional[str]
last_name: Optional[str]
email: Optional[str]
home_phone_number: Optional[str]
mobile_phone_number: Optional[str]
city: Optional[str]
state: Optional[str]
country: Optional[str]
address: Optional[str]
zip_code: Optional[str]
@property
def name(self) -> str:
"""Full name (first + last)"""
@property
def phone(self) -> Optional[str]:
"""Primary phone (mobile or home)"""
ApplicationStatus
class ApplicationStatus(Enum):
APPLIED = "applied"
SCREENING = "screening"
INTERVIEWING = "interviewing"
REFERENCE_CHECK = "reference_check"
OFFER_EXTENDED = "offer_extended"
HIRED = "hired"
REJECTED = "rejected"
WITHDRAWN = "withdrawn"
ON_HOLD = "on_hold"
UNKNOWN = "unknown" # Fallback for unrecognized values
Note: The enum supports legacy values "interview" and "offer" for backwards compatibility, automatically mapping them to INTERVIEWING and OFFER_EXTENDED respectively.
Error Handling
from apploi_partner_api import (
ApploiValidationError,
ApploiAuthenticationError,
ApploiAPIError
)
try:
job_applications = client.get_job_applications(limit=50)
except ApploiValidationError as e:
# Invalid parameters (e.g., limit > 1000)
print(f"Validation error: {e}")
except ApploiAuthenticationError as e:
# 401/403 errors
print(f"Auth failed: {e}")
except ApploiAPIError as e:
# Other API errors
print(f"API error: {e}")
if e.status_code == 429:
# Handle rate limiting
time.sleep(60)
Usage Examples
Basic Usage
# Get all job applications
all_job_applications = client.get_job_applications()
print(f"Found {len(all_job_applications)} job applications")
Filtering
# Filter with native types
filtered_job_applications = client.get_job_applications(
job_id=12345, # Integer
team_id=789, # Integer
updated_after=date(2024, 1, 1), # Date object
state="applied", # String
limit=50 # Integer
)
Pagination
# Paginate through all results
all_results = []
offset = 0
page_size = 50
while True:
page = client.get_job_applications(
limit=page_size,
offset=offset
)
all_results.extend(page.items)
if not page.has_more:
break
offset += page_size
print(f"Total fetched: {len(all_results)}")
Working with Results
job_applications = client.get_job_applications(limit=1)
if job_applications:
application = job_applications[0]
# Strongly typed fields
print(f"ID: {application.id}")
print(f"Name: {application.candidate.name}")
print(f"Status: {application.status.value}")
# Date handling
if application.applied_at:
days_ago = (datetime.now() - application.applied_at).days
print(f"Applied {days_ago} days ago")
# Access raw data if needed
raw = application.raw_data
Requirements
- Python 3.7+
- httpx (for Fern SDK)
- python-dateutil (optional, for date parsing)
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file apploi_partner_api-0.1.8.tar.gz.
File metadata
- Download URL: apploi_partner_api-0.1.8.tar.gz
- Upload date:
- Size: 5.1 MB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
63e7fe1949a120d34035b49118b65a3928c976daee4480d77205604448a0bfeb
|
|
| MD5 |
603b5ed9d893b071a2a954a2835fbbb1
|
|
| BLAKE2b-256 |
be7aacb842db3cc38d08a4dc598995facf08b5bd965271c56dbde770a733df9c
|
File details
Details for the file apploi_partner_api-0.1.8-py3-none-any.whl.
File metadata
- Download URL: apploi_partner_api-0.1.8-py3-none-any.whl
- Upload date:
- Size: 31.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
119336ac1fbbdd4b63a7231dbf35dca8c68fc246643aa30c29dcdf07d4ad6627
|
|
| MD5 |
564f41d70cb6fcb5999bf5a872bd3b65
|
|
| BLAKE2b-256 |
a6d70e31d655227cb9ab352c65af1fdb573c863f61ea39cdf2eeeaf7a684accc
|