{"openapi":"3.1.0","info":{"title":"POST /api/v1/contracts/inbound/hours","version":"1.0.0","description":"Ingest an inbound hours submission (Slack/email reply)"},"servers":[{"url":"https://api.ondayzero.com","description":"Production"}],"paths":{"/api/v1/contracts/inbound/hours":{"post":{"tags":["contracts"],"summary":"Ingest an inbound hours submission (Slack/email reply)","description":"Turn a parsed Slack or email hours reply into a pending drawdown entry awaiting approval (the hours gate). The caller supplies the already-parsed fields; the target contract must belong to the business.","operationId":"ingest_hours_submission_api_v1_contracts_inbound_hours_post","parameters":[{"name":"authorization","in":"header","required":false,"schema":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Authorization"}},{"name":"x-business-id","in":"header","required":false,"schema":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"X-Business-Id"}}],"requestBody":{"required":true,"content":{"application/json":{"schema":{"$ref":"#/components/schemas/HoursSubmissionRequest"}}}},"responses":{"201":{"description":"Successful Response","content":{"application/json":{"schema":{"$ref":"#/components/schemas/SuccessEnvelope_DrawdownResponse_"}}}},"400":{"description":"Bad Request - Invalid input","content":{"application/json":{"example":{"detail":"Invalid request parameters"}}}},"401":{"description":"Unauthorized - Authentication required","content":{"application/json":{"example":{"detail":"Not authenticated"}}}},"403":{"description":"Forbidden - Insufficient permissions","content":{"application/json":{"example":{"detail":"Not enough permissions"}}}},"404":{"description":"Not Found - Resource does not exist","content":{"application/json":{"example":{"detail":"Resource not found"}}}},"422":{"description":"Validation Error","content":{"application/json":{"schema":{"$ref":"#/components/schemas/HTTPValidationError"}}}}}}}},"components":{"schemas":{"DrawdownResponse":{"properties":{"id":{"type":"string","title":"Id"},"business_id":{"type":"string","title":"Business Id"},"contract_id":{"type":"string","title":"Contract Id"},"order_line_id":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Order Line Id"},"catalog_item_id":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Catalog Item Id"},"catalog_item_name":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Catalog Item Name"},"hours":{"type":"string","pattern":"^(?!^[-+.]*$)[+-]?0*\\d*\\.?\\d*$","title":"Hours"},"original_hours":{"anyOf":[{"type":"string","pattern":"^(?!^[-+.]*$)[+-]?0*\\d*\\.?\\d*$"},{"type":"null"}],"title":"Original Hours"},"was_edited":{"type":"boolean","title":"Was Edited","default":false},"entry_date":{"type":"string","format":"date","title":"Entry Date"},"description":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Description"},"status":{"type":"string","title":"Status"},"submitted_by":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Submitted By"},"submitted_via":{"type":"string","title":"Submitted Via"},"source_message_id":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Source Message Id"},"approved_by":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Approved By"},"approved_at":{"anyOf":[{"type":"string","format":"date-time"},{"type":"null"}],"title":"Approved At"},"created_at":{"type":"string","format":"date-time","title":"Created At"},"updated_at":{"type":"string","format":"date-time","title":"Updated At"},"can_be_approved":{"type":"boolean","title":"Can Be Approved","default":false},"can_be_returned":{"type":"boolean","title":"Can Be Returned","default":false}},"type":"object","required":["id","business_id","contract_id","hours","entry_date","status","submitted_via","created_at","updated_at"],"title":"DrawdownResponse","description":"Response schema for a drawdown entry."},"HTTPValidationError":{"properties":{"detail":{"items":{"$ref":"#/components/schemas/ValidationError"},"type":"array","title":"Detail"}},"type":"object","title":"HTTPValidationError"},"HoursSubmissionRequest":{"properties":{"contract_id":{"type":"string","title":"Contract Id","description":"Contract the hours draw against."},"hours":{"anyOf":[{"type":"number","exclusiveMinimum":0.0},{"type":"string","pattern":"^(?!^[-+.]*$)[+-]?0*\\d*\\.?\\d*$"}],"title":"Hours","description":"Hours consumed."},"entry_date":{"type":"string","title":"Entry Date","description":"Date the work happened (YYYY-MM-DD)."},"submitted_by":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Submitted By","description":"Who submitted the hours (Slack user / email sender)."},"submitted_via":{"type":"string","title":"Submitted Via","description":"Inbound channel: slack, email, or manual.","default":"slack"},"source_message_id":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Source Message Id","description":"ID of the Slack/email message that created this entry."},"description":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Description","description":"Optional note."},"order_line_id":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Order Line Id","description":"Order line the hours are drawn against (optional)."}},"type":"object","required":["contract_id","hours","entry_date"],"title":"HoursSubmissionRequest","description":"A parsed hours submission (Slack/email reply) → pending drawdown.\n\nThe Slack/email adapter verifies its own webhook signature, parses the reply,\nresolves the target contract, and posts these already-parsed fields here."},"SuccessEnvelope_DrawdownResponse_":{"properties":{"success":{"type":"boolean","title":"Success","default":true},"message":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Message"},"code":{"anyOf":[{"type":"string"},{"type":"null"}],"title":"Code"},"data":{"anyOf":[{"$ref":"#/components/schemas/DrawdownResponse"},{"type":"null"}]}},"additionalProperties":true,"type":"object","title":"SuccessEnvelope[DrawdownResponse]"},"ValidationError":{"properties":{"loc":{"items":{"anyOf":[{"type":"string"},{"type":"integer"}]},"type":"array","title":"Location"},"msg":{"type":"string","title":"Message"},"type":{"type":"string","title":"Error Type"},"input":{"title":"Input"},"ctx":{"type":"object","title":"Context"}},"type":"object","required":["loc","msg","type"],"title":"ValidationError"}},"securitySchemes":{"BearerAuth":{"type":"http","scheme":"bearer","bearerFormat":"API Token","description":"API token authentication. Format: `Bearer dz_...`"}}},"security":[{"BearerAuth":[]}]}