repull 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (222) hide show
  1. checksums.yaml +7 -0
  2. data/Gemfile +10 -0
  3. data/LICENSE +21 -0
  4. data/README.md +121 -0
  5. data/Rakefile +10 -0
  6. data/examples/connect_airbnb.rb +33 -0
  7. data/examples/quickstart.rb +22 -0
  8. data/lib/repull/api/ai_api.rb +86 -0
  9. data/lib/repull/api/airbnb_api.rb +1155 -0
  10. data/lib/repull/api/atlas_api.rb +194 -0
  11. data/lib/repull/api/availability_api.rb +167 -0
  12. data/lib/repull/api/billing_api.rb +139 -0
  13. data/lib/repull/api/booking_com_api.rb +617 -0
  14. data/lib/repull/api/connect_api.rb +672 -0
  15. data/lib/repull/api/conversations_api.rb +263 -0
  16. data/lib/repull/api/guests_api.rb +171 -0
  17. data/lib/repull/api/listings_api.rb +509 -0
  18. data/lib/repull/api/markets_api.rb +320 -0
  19. data/lib/repull/api/plumguide_api.rb +299 -0
  20. data/lib/repull/api/pricing_api.rb +453 -0
  21. data/lib/repull/api/properties_api.rb +166 -0
  22. data/lib/repull/api/reservations_api.rb +387 -0
  23. data/lib/repull/api/reviews_api.rb +208 -0
  24. data/lib/repull/api/schema_api.rb +347 -0
  25. data/lib/repull/api/system_api.rb +79 -0
  26. data/lib/repull/api/vrbo_api.rb +262 -0
  27. data/lib/repull/api/webhooks_api.rb +867 -0
  28. data/lib/repull/api_client.rb +397 -0
  29. data/lib/repull/api_error.rb +58 -0
  30. data/lib/repull/api_model_base.rb +88 -0
  31. data/lib/repull/configuration.rb +313 -0
  32. data/lib/repull/models/ai_operation.rb +158 -0
  33. data/lib/repull/models/airbnb_listing.rb +229 -0
  34. data/lib/repull/models/airbnb_listing_list_response.rb +158 -0
  35. data/lib/repull/models/airbnb_reservation.rb +224 -0
  36. data/lib/repull/models/airbnb_reservation_list_response.rb +158 -0
  37. data/lib/repull/models/airbnb_review.rb +222 -0
  38. data/lib/repull/models/airbnb_review_list_response.rb +158 -0
  39. data/lib/repull/models/airbnb_thread.rb +188 -0
  40. data/lib/repull/models/airbnb_thread_list_response.rb +158 -0
  41. data/lib/repull/models/booking_connect_listing_option.rb +201 -0
  42. data/lib/repull/models/booking_connect_room.rb +251 -0
  43. data/lib/repull/models/booking_connect_rooms_response.rb +274 -0
  44. data/lib/repull/models/booking_conversation.rb +178 -0
  45. data/lib/repull/models/booking_conversation_list_response.rb +158 -0
  46. data/lib/repull/models/booking_pricing_rate_update.rb +310 -0
  47. data/lib/repull/models/booking_pricing_rate_update_date_range.rb +190 -0
  48. data/lib/repull/models/booking_pricing_rate_update_restrictions.rb +179 -0
  49. data/lib/repull/models/booking_pricing_response.rb +157 -0
  50. data/lib/repull/models/booking_pricing_update_request.rb +176 -0
  51. data/lib/repull/models/booking_pricing_update_response.rb +199 -0
  52. data/lib/repull/models/booking_property.rb +188 -0
  53. data/lib/repull/models/booking_property_list_response.rb +158 -0
  54. data/lib/repull/models/booking_room_mapping.rb +177 -0
  55. data/lib/repull/models/booking_verify_hotel_request.rb +193 -0
  56. data/lib/repull/models/booking_verify_hotel_response.rb +285 -0
  57. data/lib/repull/models/bulk_pricing_failure.rb +177 -0
  58. data/lib/repull/models/bulk_pricing_item.rb +202 -0
  59. data/lib/repull/models/bulk_pricing_request.rb +212 -0
  60. data/lib/repull/models/bulk_pricing_response.rb +170 -0
  61. data/lib/repull/models/calendar_day.rb +174 -0
  62. data/lib/repull/models/calendar_response.rb +149 -0
  63. data/lib/repull/models/connect_host.rb +194 -0
  64. data/lib/repull/models/connect_provider.rb +365 -0
  65. data/lib/repull/models/connect_provider_list_response.rb +149 -0
  66. data/lib/repull/models/connect_session.rb +228 -0
  67. data/lib/repull/models/connect_status.rb +207 -0
  68. data/lib/repull/models/connection.rb +195 -0
  69. data/lib/repull/models/connection_list_response.rb +158 -0
  70. data/lib/repull/models/conversation.rb +257 -0
  71. data/lib/repull/models/conversation_detail.rb +284 -0
  72. data/lib/repull/models/conversation_guest.rb +178 -0
  73. data/lib/repull/models/conversation_guest_contact.rb +174 -0
  74. data/lib/repull/models/conversation_host.rb +186 -0
  75. data/lib/repull/models/conversation_list_response.rb +159 -0
  76. data/lib/repull/models/conversation_message_attachment.rb +174 -0
  77. data/lib/repull/models/create_ai_operation200_response.rb +156 -0
  78. data/lib/repull/models/create_billing_checkout_request.rb +181 -0
  79. data/lib/repull/models/create_connect_session_request.rb +189 -0
  80. data/lib/repull/models/create_connection_request.rb +225 -0
  81. data/lib/repull/models/create_reservation_request.rb +313 -0
  82. data/lib/repull/models/create_webhook_request.rb +211 -0
  83. data/lib/repull/models/custom_schema.rb +303 -0
  84. data/lib/repull/models/custom_schema_create.rb +253 -0
  85. data/lib/repull/models/custom_schema_create_response.rb +300 -0
  86. data/lib/repull/models/custom_schema_delete_response.rb +164 -0
  87. data/lib/repull/models/custom_schema_list_response.rb +159 -0
  88. data/lib/repull/models/custom_schema_summary.rb +300 -0
  89. data/lib/repull/models/custom_schema_update.rb +199 -0
  90. data/lib/repull/models/error.rb +165 -0
  91. data/lib/repull/models/error_error.rb +357 -0
  92. data/lib/repull/models/error_error_support.rb +167 -0
  93. data/lib/repull/models/get_health200_response.rb +156 -0
  94. data/lib/repull/models/guest.rb +271 -0
  95. data/lib/repull/models/guest_contact.rb +186 -0
  96. data/lib/repull/models/guest_flag.rb +179 -0
  97. data/lib/repull/models/guest_list_response.rb +159 -0
  98. data/lib/repull/models/guest_note.rb +187 -0
  99. data/lib/repull/models/guest_profile.rb +337 -0
  100. data/lib/repull/models/guest_reservations_summary.rb +175 -0
  101. data/lib/repull/models/listing.rb +216 -0
  102. data/lib/repull/models/listing_address.rb +158 -0
  103. data/lib/repull/models/listing_channel.rb +176 -0
  104. data/lib/repull/models/listing_comp.rb +272 -0
  105. data/lib/repull/models/listing_comp_nightly.rb +166 -0
  106. data/lib/repull/models/listing_comp_ratings.rb +157 -0
  107. data/lib/repull/models/listing_comps_response.rb +197 -0
  108. data/lib/repull/models/listing_content.rb +249 -0
  109. data/lib/repull/models/listing_create_request.rb +364 -0
  110. data/lib/repull/models/listing_create_response.rb +148 -0
  111. data/lib/repull/models/listing_generate_content_request.rb +192 -0
  112. data/lib/repull/models/listing_generate_content_response.rb +165 -0
  113. data/lib/repull/models/listing_list_response.rb +158 -0
  114. data/lib/repull/models/listing_pricing_apply_request.rb +193 -0
  115. data/lib/repull/models/listing_pricing_apply_response.rb +169 -0
  116. data/lib/repull/models/listing_pricing_history_entry.rb +212 -0
  117. data/lib/repull/models/listing_pricing_history_response.rb +159 -0
  118. data/lib/repull/models/listing_pricing_recommendation.rb +269 -0
  119. data/lib/repull/models/listing_pricing_response.rb +187 -0
  120. data/lib/repull/models/listing_pricing_response_comp_summary.rb +178 -0
  121. data/lib/repull/models/listing_pricing_response_date_range.rb +156 -0
  122. data/lib/repull/models/listing_pricing_response_listing.rb +191 -0
  123. data/lib/repull/models/listing_pricing_strategy.rb +317 -0
  124. data/lib/repull/models/listing_pricing_strategy_input.rb +264 -0
  125. data/lib/repull/models/listing_publish_airbnb_request.rb +171 -0
  126. data/lib/repull/models/listing_publish_response.rb +166 -0
  127. data/lib/repull/models/listing_publish_status_channel.rb +197 -0
  128. data/lib/repull/models/listing_publish_status_response.rb +158 -0
  129. data/lib/repull/models/listing_quality_tier.rb +175 -0
  130. data/lib/repull/models/listing_segment.rb +229 -0
  131. data/lib/repull/models/listing_segment_recommendation.rb +170 -0
  132. data/lib/repull/models/listing_segments_response.rb +230 -0
  133. data/lib/repull/models/listing_segments_response_scope.rb +159 -0
  134. data/lib/repull/models/map_connect_booking_rooms_request.rb +202 -0
  135. data/lib/repull/models/map_connect_booking_rooms_response.rb +243 -0
  136. data/lib/repull/models/market_browse_category.rb +158 -0
  137. data/lib/repull/models/market_browse_entry.rb +197 -0
  138. data/lib/repull/models/market_browse_featured.rb +188 -0
  139. data/lib/repull/models/market_browse_response.rb +158 -0
  140. data/lib/repull/models/market_calendar_day.rb +257 -0
  141. data/lib/repull/models/market_calendar_day_events_inner.rb +177 -0
  142. data/lib/repull/models/market_calendar_response.rb +178 -0
  143. data/lib/repull/models/market_detail_response.rb +269 -0
  144. data/lib/repull/models/market_detail_response_price_distribution_inner.rb +186 -0
  145. data/lib/repull/models/market_detail_response_property_type_mix_inner.rb +166 -0
  146. data/lib/repull/models/market_detail_response_supply_trend_inner.rb +156 -0
  147. data/lib/repull/models/market_detail_response_top_comps.rb +187 -0
  148. data/lib/repull/models/market_event.rb +257 -0
  149. data/lib/repull/models/market_my_listing.rb +242 -0
  150. data/lib/repull/models/market_summary.rb +259 -0
  151. data/lib/repull/models/market_top_comp.rb +275 -0
  152. data/lib/repull/models/markets_overview_response.rb +219 -0
  153. data/lib/repull/models/markets_overview_response_browse.rb +173 -0
  154. data/lib/repull/models/markets_overview_response_subscriptions.rb +157 -0
  155. data/lib/repull/models/markets_overview_response_totals.rb +165 -0
  156. data/lib/repull/models/message.rb +289 -0
  157. data/lib/repull/models/message_list_response.rb +159 -0
  158. data/lib/repull/models/pagination.rb +188 -0
  159. data/lib/repull/models/plumguide_listing.rb +167 -0
  160. data/lib/repull/models/plumguide_listing_list_response.rb +158 -0
  161. data/lib/repull/models/property.rb +271 -0
  162. data/lib/repull/models/property_list_response.rb +158 -0
  163. data/lib/repull/models/reservation.rb +457 -0
  164. data/lib/repull/models/reservation_list_response.rb +159 -0
  165. data/lib/repull/models/review.rb +357 -0
  166. data/lib/repull/models/review_category.rb +169 -0
  167. data/lib/repull/models/review_list_response.rb +159 -0
  168. data/lib/repull/models/review_response.rb +158 -0
  169. data/lib/repull/models/rotate_webhook_secret200_response.rb +165 -0
  170. data/lib/repull/models/select_connect_provider_request.rb +165 -0
  171. data/lib/repull/models/select_provider_response.rb +176 -0
  172. data/lib/repull/models/test_webhook_request.rb +165 -0
  173. data/lib/repull/models/update_availability_request.rb +149 -0
  174. data/lib/repull/models/update_listing_pricing_strategy200_response.rb +147 -0
  175. data/lib/repull/models/update_reservation_request.rb +174 -0
  176. data/lib/repull/models/update_webhook_request.rb +211 -0
  177. data/lib/repull/models/vrbo_listing.rb +167 -0
  178. data/lib/repull/models/vrbo_listing_list_response.rb +158 -0
  179. data/lib/repull/models/vrbo_reservation.rb +185 -0
  180. data/lib/repull/models/vrbo_reservation_list_response.rb +158 -0
  181. data/lib/repull/models/webhook_delivery.rb +243 -0
  182. data/lib/repull/models/webhook_delivery_detail.rb +261 -0
  183. data/lib/repull/models/webhook_delivery_list_response.rb +158 -0
  184. data/lib/repull/models/webhook_event_catalog.rb +149 -0
  185. data/lib/repull/models/webhook_event_catalog_domains_inner.rb +167 -0
  186. data/lib/repull/models/webhook_event_catalog_domains_inner_events_inner.rb +183 -0
  187. data/lib/repull/models/webhook_list_response.rb +158 -0
  188. data/lib/repull/models/webhook_subscription.rb +294 -0
  189. data/lib/repull/version.rb +15 -0
  190. data/lib/repull.rb +217 -0
  191. data/openapi/v1.json +8923 -0
  192. data/repull.gemspec +43 -0
  193. data/scripts/regen.sh +55 -0
  194. data/spec/api/ai_api_spec.rb +47 -0
  195. data/spec/api/airbnb_api_spec.rb +238 -0
  196. data/spec/api/availability_api_spec.rb +62 -0
  197. data/spec/api/billing_api_spec.rb +57 -0
  198. data/spec/api/booking_com_api_spec.rb +115 -0
  199. data/spec/api/connect_api_spec.rb +82 -0
  200. data/spec/api/conversations_api_spec.rb +68 -0
  201. data/spec/api/guests_api_spec.rb +59 -0
  202. data/spec/api/plumguide_api_spec.rb +85 -0
  203. data/spec/api/properties_api_spec.rb +60 -0
  204. data/spec/api/reservations_api_spec.rb +97 -0
  205. data/spec/api/system_api_spec.rb +45 -0
  206. data/spec/api/vrbo_api_spec.rb +55 -0
  207. data/spec/api/webhooks_api_spec.rb +69 -0
  208. data/spec/models/ai_operation_spec.rb +46 -0
  209. data/spec/models/calendar_day_spec.rb +54 -0
  210. data/spec/models/connect_host_spec.rb +60 -0
  211. data/spec/models/connect_status_spec.rb +76 -0
  212. data/spec/models/connection_spec.rb +70 -0
  213. data/spec/models/conversation_spec.rb +66 -0
  214. data/spec/models/error_error_spec.rb +54 -0
  215. data/spec/models/error_spec.rb +36 -0
  216. data/spec/models/guest_spec.rb +72 -0
  217. data/spec/models/message_spec.rb +70 -0
  218. data/spec/models/property_spec.rb +114 -0
  219. data/spec/models/reservation_spec.rb +128 -0
  220. data/spec/models/webhook_subscription_spec.rb +60 -0
  221. data/spec/spec_helper.rb +111 -0
  222. metadata +337 -0
@@ -0,0 +1,294 @@
1
+ =begin
2
+ #Repull API
3
+
4
+ #The unified API for vacation rental tech. Connect to 50+ PMS platforms and 4 OTA channels through one REST API. Built-in AI operations for guest communication, pricing, and listing optimization. ## Designed for AI agents Every error response on this API includes machine-parseable fields so an LLM (Claude in MCP, Cursor, Cline, GPT, etc.) can self-recover without escalating to a human: - `error.code` — stable string identifier (e.g. `invalid_params`, `rate_limit_exceeded`) - `error.message` — human-readable cause - `error.fix` — exact recovery steps (e.g. \"Pass `check_in_after` as ISO 8601: `?check_in_after=2026-01-15`\") - `error.docs_url` — link to the canonical write-up at `https://repull.dev/docs/errors/{code}` - `error.request_id` — id to correlate with server-side logs - `error.field` / `error.value_received` / `error.valid_values` / `error.did_you_mean` — when the error is parameter-specific - `error.retry_after` — seconds to wait before retrying (rate-limit + transient upstream) `Access-Control-Expose-Headers` lists `x-request-id` and the `X-RateLimit-*` family so browsers can read them on cross-origin responses. ## Quick Start 1. Get an API key at https://repull.dev/dashboard 2. Connect a PMS: `POST /v1/connect/{provider}` 3. List properties: `GET /v1/properties` 4. Get reservations: `GET /v1/reservations` ## Authentication All requests require a Bearer token: ``` Authorization: Bearer sk_test_YOUR_API_KEY ``` Sandbox keys start with `sk_test_`, production with `sk_live_`. ## Request Correlation (X-Request-ID) Every response carries an `X-Request-ID` header, e.g. `X-Request-ID: req_01HXY...`. Include this id in support tickets and bug reports — we can trace the full request lifecycle (auth, rate limit, handler, downstream calls, log row) from a single id. You may set the header on the inbound request to forward your own trace id; we will echo it back instead of generating a new one. Accepted format: `^[\\\\w.-]{1,128}$`. The id is also embedded in error envelopes as `request_id` so server-side log diffs work even when the response headers are stripped by an intermediate proxy. ## Rate Limits The public API enforces a per-API-key sliding-window rate limit on top of the per-tier monthly + daily-AI quotas. **Default policy:** 600 requests per 60 seconds, per API key. Sliding window — there is no fixed-minute boundary you can burst across. Every response includes: | Header | Meaning | |---|---| | `X-RateLimit-Limit` | Requests permitted in the current window. | | `X-RateLimit-Remaining` | Requests left in the current window after this call. | | `X-RateLimit-Reset` | Unix epoch (seconds) when the next slot opens. | | `X-RateLimit-Policy` | Machine-readable policy descriptor, e.g. `600;w=60`. | | `Retry-After` | Seconds to wait before retrying. **Only present on 429 responses.** | **On 429 (rate_limit_exceeded):** the response body matches the standard error envelope with `code: \"rate_limit_exceeded\"`, plus `limit`, `window_seconds`, `retry_after`, and `request_id` fields. SDKs MUST honor `Retry-After` and use exponential backoff with jitter on subsequent retries — never a tight loop. Recommended backoff: ``` sleep_ms = (Retry-After * 1000) + random(0..250) ``` Monthly + daily-AI tier quotas (`free`, `starter`, `pro`, `enterprise`) are enforced separately and also surface as 429s; they include `tier`, `scope`, and `resets_at` fields.
5
+
6
+ The version of the OpenAPI document: 1.0.0
7
+ Contact: ivan@vanio.ai
8
+ Generated by: https://openapi-generator.tech
9
+ Generator version: 7.22.0
10
+
11
+ =end
12
+
13
+ require 'date'
14
+ require 'time'
15
+
16
+ module Repull
17
+ # A registered webhook endpoint. The `secret` field is only present in the response of `POST /v1/webhooks` and `POST /v1/webhooks/{id}/rotate-secret` (Stripe pattern — capture it then; it is masked everywhere else).
18
+ class WebhookSubscription < ApiModelBase
19
+ attr_accessor :id
20
+
21
+ attr_accessor :url
22
+
23
+ attr_accessor :description
24
+
25
+ attr_accessor :events
26
+
27
+ attr_accessor :api_version
28
+
29
+ attr_accessor :status
30
+
31
+ attr_accessor :consecutive_failures
32
+
33
+ attr_accessor :last_delivered_at
34
+
35
+ attr_accessor :last_success_at
36
+
37
+ attr_accessor :last_failure_at
38
+
39
+ attr_accessor :last_delivery_status
40
+
41
+ attr_accessor :disabled_at
42
+
43
+ attr_accessor :created_at
44
+
45
+ attr_accessor :updated_at
46
+
47
+ attr_accessor :secret_masked
48
+
49
+ # Plaintext signing secret. Only returned by create + rotate. Capture and store securely.
50
+ attr_accessor :secret
51
+
52
+ # Attribute mapping from ruby-style variable name to JSON key.
53
+ def self.attribute_map
54
+ {
55
+ :'id' => :'id',
56
+ :'url' => :'url',
57
+ :'description' => :'description',
58
+ :'events' => :'events',
59
+ :'api_version' => :'apiVersion',
60
+ :'status' => :'status',
61
+ :'consecutive_failures' => :'consecutiveFailures',
62
+ :'last_delivered_at' => :'lastDeliveredAt',
63
+ :'last_success_at' => :'lastSuccessAt',
64
+ :'last_failure_at' => :'lastFailureAt',
65
+ :'last_delivery_status' => :'lastDeliveryStatus',
66
+ :'disabled_at' => :'disabledAt',
67
+ :'created_at' => :'createdAt',
68
+ :'updated_at' => :'updatedAt',
69
+ :'secret_masked' => :'secretMasked',
70
+ :'secret' => :'secret'
71
+ }
72
+ end
73
+
74
+ # Returns attribute mapping this model knows about
75
+ def self.acceptable_attribute_map
76
+ attribute_map
77
+ end
78
+
79
+ # Returns all the JSON keys this model knows about
80
+ def self.acceptable_attributes
81
+ acceptable_attribute_map.values
82
+ end
83
+
84
+ # Attribute type mapping.
85
+ def self.openapi_types
86
+ {
87
+ :'id' => :'String',
88
+ :'url' => :'String',
89
+ :'description' => :'String',
90
+ :'events' => :'Array<String>',
91
+ :'api_version' => :'String',
92
+ :'status' => :'String',
93
+ :'consecutive_failures' => :'Integer',
94
+ :'last_delivered_at' => :'Time',
95
+ :'last_success_at' => :'Time',
96
+ :'last_failure_at' => :'Time',
97
+ :'last_delivery_status' => :'Integer',
98
+ :'disabled_at' => :'Time',
99
+ :'created_at' => :'Time',
100
+ :'updated_at' => :'Time',
101
+ :'secret_masked' => :'String',
102
+ :'secret' => :'String'
103
+ }
104
+ end
105
+
106
+ # List of attributes with nullable: true
107
+ def self.openapi_nullable
108
+ Set.new([
109
+ :'description',
110
+ :'last_delivered_at',
111
+ :'last_success_at',
112
+ :'last_failure_at',
113
+ :'last_delivery_status',
114
+ :'disabled_at',
115
+ :'secret_masked',
116
+ :'secret'
117
+ ])
118
+ end
119
+
120
+ # Initializes the object
121
+ # @param [Hash] attributes Model attributes in the form of hash
122
+ def initialize(attributes = {})
123
+ if (!attributes.is_a?(Hash))
124
+ fail ArgumentError, "The input argument (attributes) must be a hash in `Repull::WebhookSubscription` initialize method"
125
+ end
126
+
127
+ # check to see if the attribute exists and convert string to symbol for hash key
128
+ acceptable_attribute_map = self.class.acceptable_attribute_map
129
+ attributes = attributes.each_with_object({}) { |(k, v), h|
130
+ if (!acceptable_attribute_map.key?(k.to_sym))
131
+ fail ArgumentError, "`#{k}` is not a valid attribute in `Repull::WebhookSubscription`. Please check the name to make sure it's valid. List of attributes: " + acceptable_attribute_map.keys.inspect
132
+ end
133
+ h[k.to_sym] = v
134
+ }
135
+
136
+ if attributes.key?(:'id')
137
+ self.id = attributes[:'id']
138
+ end
139
+
140
+ if attributes.key?(:'url')
141
+ self.url = attributes[:'url']
142
+ end
143
+
144
+ if attributes.key?(:'description')
145
+ self.description = attributes[:'description']
146
+ end
147
+
148
+ if attributes.key?(:'events')
149
+ if (value = attributes[:'events']).is_a?(Array)
150
+ self.events = value
151
+ end
152
+ end
153
+
154
+ if attributes.key?(:'api_version')
155
+ self.api_version = attributes[:'api_version']
156
+ end
157
+
158
+ if attributes.key?(:'status')
159
+ self.status = attributes[:'status']
160
+ end
161
+
162
+ if attributes.key?(:'consecutive_failures')
163
+ self.consecutive_failures = attributes[:'consecutive_failures']
164
+ end
165
+
166
+ if attributes.key?(:'last_delivered_at')
167
+ self.last_delivered_at = attributes[:'last_delivered_at']
168
+ end
169
+
170
+ if attributes.key?(:'last_success_at')
171
+ self.last_success_at = attributes[:'last_success_at']
172
+ end
173
+
174
+ if attributes.key?(:'last_failure_at')
175
+ self.last_failure_at = attributes[:'last_failure_at']
176
+ end
177
+
178
+ if attributes.key?(:'last_delivery_status')
179
+ self.last_delivery_status = attributes[:'last_delivery_status']
180
+ end
181
+
182
+ if attributes.key?(:'disabled_at')
183
+ self.disabled_at = attributes[:'disabled_at']
184
+ end
185
+
186
+ if attributes.key?(:'created_at')
187
+ self.created_at = attributes[:'created_at']
188
+ end
189
+
190
+ if attributes.key?(:'updated_at')
191
+ self.updated_at = attributes[:'updated_at']
192
+ end
193
+
194
+ if attributes.key?(:'secret_masked')
195
+ self.secret_masked = attributes[:'secret_masked']
196
+ end
197
+
198
+ if attributes.key?(:'secret')
199
+ self.secret = attributes[:'secret']
200
+ end
201
+ end
202
+
203
+ # Show invalid properties with the reasons. Usually used together with valid?
204
+ # @return Array for valid properties with the reasons
205
+ def list_invalid_properties
206
+ warn '[DEPRECATED] the `list_invalid_properties` method is obsolete'
207
+ invalid_properties = Array.new
208
+ invalid_properties
209
+ end
210
+
211
+ # Check to see if the all the properties in the model are valid
212
+ # @return true if the model is valid
213
+ def valid?
214
+ warn '[DEPRECATED] the `valid?` method is obsolete'
215
+ true
216
+ end
217
+
218
+ # Checks equality by comparing each attribute.
219
+ # @param [Object] Object to be compared
220
+ def ==(o)
221
+ return true if self.equal?(o)
222
+ self.class == o.class &&
223
+ id == o.id &&
224
+ url == o.url &&
225
+ description == o.description &&
226
+ events == o.events &&
227
+ api_version == o.api_version &&
228
+ status == o.status &&
229
+ consecutive_failures == o.consecutive_failures &&
230
+ last_delivered_at == o.last_delivered_at &&
231
+ last_success_at == o.last_success_at &&
232
+ last_failure_at == o.last_failure_at &&
233
+ last_delivery_status == o.last_delivery_status &&
234
+ disabled_at == o.disabled_at &&
235
+ created_at == o.created_at &&
236
+ updated_at == o.updated_at &&
237
+ secret_masked == o.secret_masked &&
238
+ secret == o.secret
239
+ end
240
+
241
+ # @see the `==` method
242
+ # @param [Object] Object to be compared
243
+ def eql?(o)
244
+ self == o
245
+ end
246
+
247
+ # Calculates hash code according to all attributes.
248
+ # @return [Integer] Hash code
249
+ def hash
250
+ [id, url, description, events, api_version, status, consecutive_failures, last_delivered_at, last_success_at, last_failure_at, last_delivery_status, disabled_at, created_at, updated_at, secret_masked, secret].hash
251
+ end
252
+
253
+ # Builds the object from hash
254
+ # @param [Hash] attributes Model attributes in the form of hash
255
+ # @return [Object] Returns the model itself
256
+ def self.build_from_hash(attributes)
257
+ return nil unless attributes.is_a?(Hash)
258
+ attributes = attributes.transform_keys(&:to_sym)
259
+ transformed_hash = {}
260
+ openapi_types.each_pair do |key, type|
261
+ if attributes.key?(attribute_map[key]) && attributes[attribute_map[key]].nil?
262
+ transformed_hash["#{key}"] = nil
263
+ elsif type =~ /\AArray<(.*)>/i
264
+ # check to ensure the input is an array given that the attribute
265
+ # is documented as an array but the input is not
266
+ if attributes[attribute_map[key]].is_a?(Array)
267
+ transformed_hash["#{key}"] = attributes[attribute_map[key]].map { |v| _deserialize($1, v) }
268
+ end
269
+ elsif !attributes[attribute_map[key]].nil?
270
+ transformed_hash["#{key}"] = _deserialize(type, attributes[attribute_map[key]])
271
+ end
272
+ end
273
+ new(transformed_hash)
274
+ end
275
+
276
+ # Returns the object in the form of hash
277
+ # @return [Hash] Returns the object in the form of hash
278
+ def to_hash
279
+ hash = {}
280
+ self.class.attribute_map.each_pair do |attr, param|
281
+ value = self.send(attr)
282
+ if value.nil?
283
+ is_nullable = self.class.openapi_nullable.include?(attr)
284
+ next if !is_nullable || (is_nullable && !instance_variable_defined?(:"@#{attr}"))
285
+ end
286
+
287
+ hash[param] = _to_hash(value)
288
+ end
289
+ hash
290
+ end
291
+
292
+ end
293
+
294
+ end
@@ -0,0 +1,15 @@
1
+ =begin
2
+ #Repull API
3
+
4
+ #The unified API for vacation rental tech. Connect to 50+ PMS platforms and 4 OTA channels through one REST API. Built-in AI operations for guest communication, pricing, and listing optimization. ## Designed for AI agents Every error response on this API includes machine-parseable fields so an LLM (Claude in MCP, Cursor, Cline, GPT, etc.) can self-recover without escalating to a human: - `error.code` — stable string identifier (e.g. `invalid_params`, `rate_limit_exceeded`) - `error.message` — human-readable cause - `error.fix` — exact recovery steps (e.g. \"Pass `check_in_after` as ISO 8601: `?check_in_after=2026-01-15`\") - `error.docs_url` — link to the canonical write-up at `https://repull.dev/docs/errors/{code}` - `error.request_id` — id to correlate with server-side logs - `error.field` / `error.value_received` / `error.valid_values` / `error.did_you_mean` — when the error is parameter-specific - `error.retry_after` — seconds to wait before retrying (rate-limit + transient upstream) `Access-Control-Expose-Headers` lists `x-request-id` and the `X-RateLimit-*` family so browsers can read them on cross-origin responses. ## Quick Start 1. Get an API key at https://repull.dev/dashboard 2. Connect a PMS: `POST /v1/connect/{provider}` 3. List properties: `GET /v1/properties` 4. Get reservations: `GET /v1/reservations` ## Authentication All requests require a Bearer token: ``` Authorization: Bearer sk_test_YOUR_API_KEY ``` Sandbox keys start with `sk_test_`, production with `sk_live_`. ## Request Correlation (X-Request-ID) Every response carries an `X-Request-ID` header, e.g. `X-Request-ID: req_01HXY...`. Include this id in support tickets and bug reports — we can trace the full request lifecycle (auth, rate limit, handler, downstream calls, log row) from a single id. You may set the header on the inbound request to forward your own trace id; we will echo it back instead of generating a new one. Accepted format: `^[\\\\w.-]{1,128}$`. The id is also embedded in error envelopes as `request_id` so server-side log diffs work even when the response headers are stripped by an intermediate proxy. ## Rate Limits The public API enforces a per-API-key sliding-window rate limit on top of the per-tier monthly + daily-AI quotas. **Default policy:** 600 requests per 60 seconds, per API key. Sliding window — there is no fixed-minute boundary you can burst across. Every response includes: | Header | Meaning | |---|---| | `X-RateLimit-Limit` | Requests permitted in the current window. | | `X-RateLimit-Remaining` | Requests left in the current window after this call. | | `X-RateLimit-Reset` | Unix epoch (seconds) when the next slot opens. | | `X-RateLimit-Policy` | Machine-readable policy descriptor, e.g. `600;w=60`. | | `Retry-After` | Seconds to wait before retrying. **Only present on 429 responses.** | **On 429 (rate_limit_exceeded):** the response body matches the standard error envelope with `code: \"rate_limit_exceeded\"`, plus `limit`, `window_seconds`, `retry_after`, and `request_id` fields. SDKs MUST honor `Retry-After` and use exponential backoff with jitter on subsequent retries — never a tight loop. Recommended backoff: ``` sleep_ms = (Retry-After * 1000) + random(0..250) ``` Monthly + daily-AI tier quotas (`free`, `starter`, `pro`, `enterprise`) are enforced separately and also surface as 429s; they include `tier`, `scope`, and `resets_at` fields.
5
+
6
+ The version of the OpenAPI document: 1.0.0
7
+ Contact: ivan@vanio.ai
8
+ Generated by: https://openapi-generator.tech
9
+ Generator version: 7.22.0
10
+
11
+ =end
12
+
13
+ module Repull
14
+ VERSION = '0.2.0'
15
+ end
data/lib/repull.rb ADDED
@@ -0,0 +1,217 @@
1
+ =begin
2
+ #Repull API
3
+
4
+ #The unified API for vacation rental tech. Connect to 50+ PMS platforms and 4 OTA channels through one REST API. Built-in AI operations for guest communication, pricing, and listing optimization. ## Designed for AI agents Every error response on this API includes machine-parseable fields so an LLM (Claude in MCP, Cursor, Cline, GPT, etc.) can self-recover without escalating to a human: - `error.code` — stable string identifier (e.g. `invalid_params`, `rate_limit_exceeded`) - `error.message` — human-readable cause - `error.fix` — exact recovery steps (e.g. \"Pass `check_in_after` as ISO 8601: `?check_in_after=2026-01-15`\") - `error.docs_url` — link to the canonical write-up at `https://repull.dev/docs/errors/{code}` - `error.request_id` — id to correlate with server-side logs - `error.field` / `error.value_received` / `error.valid_values` / `error.did_you_mean` — when the error is parameter-specific - `error.retry_after` — seconds to wait before retrying (rate-limit + transient upstream) `Access-Control-Expose-Headers` lists `x-request-id` and the `X-RateLimit-*` family so browsers can read them on cross-origin responses. ## Quick Start 1. Get an API key at https://repull.dev/dashboard 2. Connect a PMS: `POST /v1/connect/{provider}` 3. List properties: `GET /v1/properties` 4. Get reservations: `GET /v1/reservations` ## Authentication All requests require a Bearer token: ``` Authorization: Bearer sk_test_YOUR_API_KEY ``` Sandbox keys start with `sk_test_`, production with `sk_live_`. ## Request Correlation (X-Request-ID) Every response carries an `X-Request-ID` header, e.g. `X-Request-ID: req_01HXY...`. Include this id in support tickets and bug reports — we can trace the full request lifecycle (auth, rate limit, handler, downstream calls, log row) from a single id. You may set the header on the inbound request to forward your own trace id; we will echo it back instead of generating a new one. Accepted format: `^[\\\\w.-]{1,128}$`. The id is also embedded in error envelopes as `request_id` so server-side log diffs work even when the response headers are stripped by an intermediate proxy. ## Rate Limits The public API enforces a per-API-key sliding-window rate limit on top of the per-tier monthly + daily-AI quotas. **Default policy:** 600 requests per 60 seconds, per API key. Sliding window — there is no fixed-minute boundary you can burst across. Every response includes: | Header | Meaning | |---|---| | `X-RateLimit-Limit` | Requests permitted in the current window. | | `X-RateLimit-Remaining` | Requests left in the current window after this call. | | `X-RateLimit-Reset` | Unix epoch (seconds) when the next slot opens. | | `X-RateLimit-Policy` | Machine-readable policy descriptor, e.g. `600;w=60`. | | `Retry-After` | Seconds to wait before retrying. **Only present on 429 responses.** | **On 429 (rate_limit_exceeded):** the response body matches the standard error envelope with `code: \"rate_limit_exceeded\"`, plus `limit`, `window_seconds`, `retry_after`, and `request_id` fields. SDKs MUST honor `Retry-After` and use exponential backoff with jitter on subsequent retries — never a tight loop. Recommended backoff: ``` sleep_ms = (Retry-After * 1000) + random(0..250) ``` Monthly + daily-AI tier quotas (`free`, `starter`, `pro`, `enterprise`) are enforced separately and also surface as 429s; they include `tier`, `scope`, and `resets_at` fields.
5
+
6
+ The version of the OpenAPI document: 1.0.0
7
+ Contact: ivan@vanio.ai
8
+ Generated by: https://openapi-generator.tech
9
+ Generator version: 7.22.0
10
+
11
+ =end
12
+
13
+ # Common files
14
+ require 'repull/api_client'
15
+ require 'repull/api_error'
16
+ require 'repull/api_model_base'
17
+ require 'repull/version'
18
+ require 'repull/configuration'
19
+
20
+ # Models
21
+ require 'repull/models/ai_operation'
22
+ require 'repull/models/airbnb_listing'
23
+ require 'repull/models/airbnb_listing_list_response'
24
+ require 'repull/models/airbnb_reservation'
25
+ require 'repull/models/airbnb_reservation_list_response'
26
+ require 'repull/models/airbnb_review'
27
+ require 'repull/models/airbnb_review_list_response'
28
+ require 'repull/models/airbnb_thread'
29
+ require 'repull/models/airbnb_thread_list_response'
30
+ require 'repull/models/booking_connect_listing_option'
31
+ require 'repull/models/booking_connect_room'
32
+ require 'repull/models/booking_connect_rooms_response'
33
+ require 'repull/models/booking_conversation'
34
+ require 'repull/models/booking_conversation_list_response'
35
+ require 'repull/models/booking_pricing_rate_update'
36
+ require 'repull/models/booking_pricing_rate_update_date_range'
37
+ require 'repull/models/booking_pricing_rate_update_restrictions'
38
+ require 'repull/models/booking_pricing_response'
39
+ require 'repull/models/booking_pricing_update_request'
40
+ require 'repull/models/booking_pricing_update_response'
41
+ require 'repull/models/booking_property'
42
+ require 'repull/models/booking_property_list_response'
43
+ require 'repull/models/booking_room_mapping'
44
+ require 'repull/models/booking_verify_hotel_request'
45
+ require 'repull/models/booking_verify_hotel_response'
46
+ require 'repull/models/bulk_pricing_failure'
47
+ require 'repull/models/bulk_pricing_item'
48
+ require 'repull/models/bulk_pricing_request'
49
+ require 'repull/models/bulk_pricing_response'
50
+ require 'repull/models/calendar_day'
51
+ require 'repull/models/calendar_response'
52
+ require 'repull/models/connect_host'
53
+ require 'repull/models/connect_provider'
54
+ require 'repull/models/connect_provider_list_response'
55
+ require 'repull/models/connect_session'
56
+ require 'repull/models/connect_status'
57
+ require 'repull/models/connection'
58
+ require 'repull/models/connection_list_response'
59
+ require 'repull/models/conversation'
60
+ require 'repull/models/conversation_detail'
61
+ require 'repull/models/conversation_guest'
62
+ require 'repull/models/conversation_guest_contact'
63
+ require 'repull/models/conversation_host'
64
+ require 'repull/models/conversation_list_response'
65
+ require 'repull/models/conversation_message_attachment'
66
+ require 'repull/models/create_ai_operation200_response'
67
+ require 'repull/models/create_billing_checkout_request'
68
+ require 'repull/models/create_connect_session_request'
69
+ require 'repull/models/create_connection_request'
70
+ require 'repull/models/create_reservation_request'
71
+ require 'repull/models/create_webhook_request'
72
+ require 'repull/models/custom_schema'
73
+ require 'repull/models/custom_schema_create'
74
+ require 'repull/models/custom_schema_create_response'
75
+ require 'repull/models/custom_schema_delete_response'
76
+ require 'repull/models/custom_schema_list_response'
77
+ require 'repull/models/custom_schema_summary'
78
+ require 'repull/models/custom_schema_update'
79
+ require 'repull/models/error'
80
+ require 'repull/models/error_error'
81
+ require 'repull/models/error_error_support'
82
+ require 'repull/models/get_health200_response'
83
+ require 'repull/models/guest'
84
+ require 'repull/models/guest_contact'
85
+ require 'repull/models/guest_flag'
86
+ require 'repull/models/guest_list_response'
87
+ require 'repull/models/guest_note'
88
+ require 'repull/models/guest_profile'
89
+ require 'repull/models/guest_reservations_summary'
90
+ require 'repull/models/listing'
91
+ require 'repull/models/listing_address'
92
+ require 'repull/models/listing_channel'
93
+ require 'repull/models/listing_comp'
94
+ require 'repull/models/listing_comp_nightly'
95
+ require 'repull/models/listing_comp_ratings'
96
+ require 'repull/models/listing_comps_response'
97
+ require 'repull/models/listing_content'
98
+ require 'repull/models/listing_create_request'
99
+ require 'repull/models/listing_create_response'
100
+ require 'repull/models/listing_generate_content_request'
101
+ require 'repull/models/listing_generate_content_response'
102
+ require 'repull/models/listing_list_response'
103
+ require 'repull/models/listing_pricing_apply_request'
104
+ require 'repull/models/listing_pricing_apply_response'
105
+ require 'repull/models/listing_pricing_history_entry'
106
+ require 'repull/models/listing_pricing_history_response'
107
+ require 'repull/models/listing_pricing_recommendation'
108
+ require 'repull/models/listing_pricing_response'
109
+ require 'repull/models/listing_pricing_response_comp_summary'
110
+ require 'repull/models/listing_pricing_response_date_range'
111
+ require 'repull/models/listing_pricing_response_listing'
112
+ require 'repull/models/listing_pricing_strategy'
113
+ require 'repull/models/listing_pricing_strategy_input'
114
+ require 'repull/models/listing_publish_airbnb_request'
115
+ require 'repull/models/listing_publish_response'
116
+ require 'repull/models/listing_publish_status_channel'
117
+ require 'repull/models/listing_publish_status_response'
118
+ require 'repull/models/listing_quality_tier'
119
+ require 'repull/models/listing_segment'
120
+ require 'repull/models/listing_segment_recommendation'
121
+ require 'repull/models/listing_segments_response'
122
+ require 'repull/models/listing_segments_response_scope'
123
+ require 'repull/models/map_connect_booking_rooms_request'
124
+ require 'repull/models/map_connect_booking_rooms_response'
125
+ require 'repull/models/market_browse_category'
126
+ require 'repull/models/market_browse_entry'
127
+ require 'repull/models/market_browse_featured'
128
+ require 'repull/models/market_browse_response'
129
+ require 'repull/models/market_calendar_day'
130
+ require 'repull/models/market_calendar_day_events_inner'
131
+ require 'repull/models/market_calendar_response'
132
+ require 'repull/models/market_detail_response'
133
+ require 'repull/models/market_detail_response_price_distribution_inner'
134
+ require 'repull/models/market_detail_response_property_type_mix_inner'
135
+ require 'repull/models/market_detail_response_supply_trend_inner'
136
+ require 'repull/models/market_detail_response_top_comps'
137
+ require 'repull/models/market_event'
138
+ require 'repull/models/market_my_listing'
139
+ require 'repull/models/market_summary'
140
+ require 'repull/models/market_top_comp'
141
+ require 'repull/models/markets_overview_response'
142
+ require 'repull/models/markets_overview_response_browse'
143
+ require 'repull/models/markets_overview_response_subscriptions'
144
+ require 'repull/models/markets_overview_response_totals'
145
+ require 'repull/models/message'
146
+ require 'repull/models/message_list_response'
147
+ require 'repull/models/pagination'
148
+ require 'repull/models/plumguide_listing'
149
+ require 'repull/models/plumguide_listing_list_response'
150
+ require 'repull/models/property'
151
+ require 'repull/models/property_list_response'
152
+ require 'repull/models/reservation'
153
+ require 'repull/models/reservation_list_response'
154
+ require 'repull/models/review'
155
+ require 'repull/models/review_category'
156
+ require 'repull/models/review_list_response'
157
+ require 'repull/models/review_response'
158
+ require 'repull/models/rotate_webhook_secret200_response'
159
+ require 'repull/models/select_connect_provider_request'
160
+ require 'repull/models/select_provider_response'
161
+ require 'repull/models/test_webhook_request'
162
+ require 'repull/models/update_availability_request'
163
+ require 'repull/models/update_listing_pricing_strategy200_response'
164
+ require 'repull/models/update_reservation_request'
165
+ require 'repull/models/update_webhook_request'
166
+ require 'repull/models/vrbo_listing'
167
+ require 'repull/models/vrbo_listing_list_response'
168
+ require 'repull/models/vrbo_reservation'
169
+ require 'repull/models/vrbo_reservation_list_response'
170
+ require 'repull/models/webhook_delivery'
171
+ require 'repull/models/webhook_delivery_detail'
172
+ require 'repull/models/webhook_delivery_list_response'
173
+ require 'repull/models/webhook_event_catalog'
174
+ require 'repull/models/webhook_event_catalog_domains_inner'
175
+ require 'repull/models/webhook_event_catalog_domains_inner_events_inner'
176
+ require 'repull/models/webhook_list_response'
177
+ require 'repull/models/webhook_subscription'
178
+
179
+ # APIs
180
+ require 'repull/api/ai_api'
181
+ require 'repull/api/airbnb_api'
182
+ require 'repull/api/atlas_api'
183
+ require 'repull/api/availability_api'
184
+ require 'repull/api/billing_api'
185
+ require 'repull/api/booking_com_api'
186
+ require 'repull/api/connect_api'
187
+ require 'repull/api/conversations_api'
188
+ require 'repull/api/guests_api'
189
+ require 'repull/api/listings_api'
190
+ require 'repull/api/markets_api'
191
+ require 'repull/api/plumguide_api'
192
+ require 'repull/api/pricing_api'
193
+ require 'repull/api/properties_api'
194
+ require 'repull/api/reservations_api'
195
+ require 'repull/api/reviews_api'
196
+ require 'repull/api/schema_api'
197
+ require 'repull/api/system_api'
198
+ require 'repull/api/vrbo_api'
199
+ require 'repull/api/webhooks_api'
200
+
201
+ module Repull
202
+ class << self
203
+ # Customize default settings for the SDK using block.
204
+ # Repull.configure do |config|
205
+ # config.username = "xxx"
206
+ # config.password = "xxx"
207
+ # end
208
+ # If no block given, return the default Configuration object.
209
+ def configure
210
+ if block_given?
211
+ yield(Configuration.default)
212
+ else
213
+ Configuration.default
214
+ end
215
+ end
216
+ end
217
+ end