firecrawl 1.4.0__tar.gz → 1.5.0__tar.gz
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.
Potentially problematic release.
This version of firecrawl might be problematic. Click here for more details.
- {firecrawl-1.4.0 → firecrawl-1.5.0}/PKG-INFO +3 -3
- {firecrawl-1.4.0 → firecrawl-1.5.0}/README.md +2 -2
- firecrawl-1.5.0/firecrawl/__init__.py +79 -0
- firecrawl-1.5.0/firecrawl/__tests__/e2e_withAuth/__init__.py +0 -0
- firecrawl-1.5.0/firecrawl/__tests__/e2e_withAuth/test.py +170 -0
- firecrawl-1.5.0/firecrawl/__tests__/v1/e2e_withAuth/__init__.py +0 -0
- firecrawl-1.5.0/firecrawl/__tests__/v1/e2e_withAuth/test.py +352 -0
- {firecrawl-1.4.0 → firecrawl-1.5.0}/firecrawl/firecrawl.py +60 -18
- {firecrawl-1.4.0 → firecrawl-1.5.0}/firecrawl.egg-info/PKG-INFO +3 -3
- {firecrawl-1.4.0 → firecrawl-1.5.0}/firecrawl.egg-info/SOURCES.txt +5 -1
- firecrawl-1.5.0/firecrawl.egg-info/top_level.txt +1 -0
- firecrawl-1.4.0/firecrawl/__init__.py +0 -57
- firecrawl-1.4.0/firecrawl.egg-info/top_level.txt +0 -3
- {firecrawl-1.4.0 → firecrawl-1.5.0}/LICENSE +0 -0
- {firecrawl-1.4.0 → firecrawl-1.5.0}/firecrawl.egg-info/dependency_links.txt +0 -0
- {firecrawl-1.4.0 → firecrawl-1.5.0}/firecrawl.egg-info/requires.txt +0 -0
- {firecrawl-1.4.0 → firecrawl-1.5.0}/pyproject.toml +0 -0
- {firecrawl-1.4.0 → firecrawl-1.5.0}/setup.cfg +0 -0
- {firecrawl-1.4.0 → firecrawl-1.5.0}/setup.py +0 -0
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: firecrawl
|
|
3
|
-
Version: 1.
|
|
3
|
+
Version: 1.5.0
|
|
4
4
|
Summary: Python SDK for Firecrawl API
|
|
5
5
|
Home-page: https://github.com/mendableai/firecrawl
|
|
6
6
|
Author: Mendable.ai
|
|
@@ -210,11 +210,11 @@ print(batch_scrape_result)
|
|
|
210
210
|
|
|
211
211
|
### Checking batch scrape status
|
|
212
212
|
|
|
213
|
-
To check the status of an asynchronous batch scrape job, use the `
|
|
213
|
+
To check the status of an asynchronous batch scrape job, use the `check_batch_scrape_status` method. It takes the job ID as a parameter and returns the current status of the batch scrape job.
|
|
214
214
|
|
|
215
215
|
```python
|
|
216
216
|
id = batch_scrape_result['id']
|
|
217
|
-
status = app.
|
|
217
|
+
status = app.check_batch_scrape_status(id)
|
|
218
218
|
```
|
|
219
219
|
|
|
220
220
|
### Batch scrape with WebSockets
|
|
@@ -170,11 +170,11 @@ print(batch_scrape_result)
|
|
|
170
170
|
|
|
171
171
|
### Checking batch scrape status
|
|
172
172
|
|
|
173
|
-
To check the status of an asynchronous batch scrape job, use the `
|
|
173
|
+
To check the status of an asynchronous batch scrape job, use the `check_batch_scrape_status` method. It takes the job ID as a parameter and returns the current status of the batch scrape job.
|
|
174
174
|
|
|
175
175
|
```python
|
|
176
176
|
id = batch_scrape_result['id']
|
|
177
|
-
status = app.
|
|
177
|
+
status = app.check_batch_scrape_status(id)
|
|
178
178
|
```
|
|
179
179
|
|
|
180
180
|
### Batch scrape with WebSockets
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
"""
|
|
2
|
+
This is the Firecrawl package.
|
|
3
|
+
|
|
4
|
+
This package provides a Python SDK for interacting with the Firecrawl API.
|
|
5
|
+
It includes methods to scrape URLs, perform searches, initiate and monitor crawl jobs,
|
|
6
|
+
and check the status of these jobs.
|
|
7
|
+
|
|
8
|
+
For more information visit https://github.com/firecrawl/
|
|
9
|
+
"""
|
|
10
|
+
|
|
11
|
+
import logging
|
|
12
|
+
import os
|
|
13
|
+
|
|
14
|
+
from .firecrawl import FirecrawlApp # noqa
|
|
15
|
+
|
|
16
|
+
__version__ = "1.5.0"
|
|
17
|
+
|
|
18
|
+
# Define the logger for the Firecrawl project
|
|
19
|
+
logger: logging.Logger = logging.getLogger("firecrawl")
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def _configure_logger() -> None:
|
|
23
|
+
"""
|
|
24
|
+
Configure the firecrawl logger for console output.
|
|
25
|
+
|
|
26
|
+
The function attaches a handler for console output with a specific format and date
|
|
27
|
+
format to the firecrawl logger.
|
|
28
|
+
"""
|
|
29
|
+
try:
|
|
30
|
+
# Create the formatter
|
|
31
|
+
formatter = logging.Formatter(
|
|
32
|
+
"[%(asctime)s - %(name)s:%(lineno)d - %(levelname)s] %(message)s",
|
|
33
|
+
datefmt="%Y-%m-%d %H:%M:%S",
|
|
34
|
+
)
|
|
35
|
+
|
|
36
|
+
# Create the console handler and set the formatter
|
|
37
|
+
console_handler = logging.StreamHandler()
|
|
38
|
+
console_handler.setFormatter(formatter)
|
|
39
|
+
|
|
40
|
+
# Add the console handler to the firecrawl logger
|
|
41
|
+
logger.addHandler(console_handler)
|
|
42
|
+
except Exception as e:
|
|
43
|
+
logger.error("Failed to configure logging: %s", e)
|
|
44
|
+
|
|
45
|
+
|
|
46
|
+
def setup_logging() -> None:
|
|
47
|
+
"""Set up logging based on the FIRECRAWL_LOGGING_LEVEL environment variable."""
|
|
48
|
+
# Check if the firecrawl logger already has a handler
|
|
49
|
+
if logger.hasHandlers():
|
|
50
|
+
return # To prevent duplicate logging
|
|
51
|
+
|
|
52
|
+
# Check if the FIRECRAWL_LOGGING_LEVEL environment variable is set
|
|
53
|
+
if not (env := os.getenv("FIRECRAWL_LOGGING_LEVEL", "").upper()):
|
|
54
|
+
# Attach a no-op handler to prevent warnings about no handlers
|
|
55
|
+
logger.addHandler(logging.NullHandler())
|
|
56
|
+
return
|
|
57
|
+
|
|
58
|
+
# Attach the console handler to the firecrawl logger
|
|
59
|
+
_configure_logger()
|
|
60
|
+
|
|
61
|
+
# Set the logging level based on the FIRECRAWL_LOGGING_LEVEL environment variable
|
|
62
|
+
if env == "DEBUG":
|
|
63
|
+
logger.setLevel(logging.DEBUG)
|
|
64
|
+
elif env == "INFO":
|
|
65
|
+
logger.setLevel(logging.INFO)
|
|
66
|
+
elif env == "WARNING":
|
|
67
|
+
logger.setLevel(logging.WARNING)
|
|
68
|
+
elif env == "ERROR":
|
|
69
|
+
logger.setLevel(logging.ERROR)
|
|
70
|
+
elif env == "CRITICAL":
|
|
71
|
+
logger.setLevel(logging.CRITICAL)
|
|
72
|
+
else:
|
|
73
|
+
logger.setLevel(logging.INFO)
|
|
74
|
+
logger.warning("Unknown logging level: %s, defaulting to INFO", env)
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
# Initialize logging configuration when the module is imported
|
|
78
|
+
setup_logging()
|
|
79
|
+
logger.debug("Debugging logger setup")
|
|
File without changes
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import importlib.util
|
|
2
|
+
import pytest
|
|
3
|
+
import time
|
|
4
|
+
import os
|
|
5
|
+
from uuid import uuid4
|
|
6
|
+
from dotenv import load_dotenv
|
|
7
|
+
|
|
8
|
+
load_dotenv()
|
|
9
|
+
|
|
10
|
+
API_URL = "http://127.0.0.1:3002"
|
|
11
|
+
ABSOLUTE_FIRECRAWL_PATH = "firecrawl/firecrawl.py"
|
|
12
|
+
TEST_API_KEY = os.getenv('TEST_API_KEY')
|
|
13
|
+
|
|
14
|
+
print(f"ABSOLUTE_FIRECRAWL_PATH: {ABSOLUTE_FIRECRAWL_PATH}")
|
|
15
|
+
|
|
16
|
+
spec = importlib.util.spec_from_file_location("FirecrawlApp", ABSOLUTE_FIRECRAWL_PATH)
|
|
17
|
+
firecrawl = importlib.util.module_from_spec(spec)
|
|
18
|
+
spec.loader.exec_module(firecrawl)
|
|
19
|
+
FirecrawlApp = firecrawl.FirecrawlApp
|
|
20
|
+
|
|
21
|
+
def test_no_api_key():
|
|
22
|
+
with pytest.raises(Exception) as excinfo:
|
|
23
|
+
invalid_app = FirecrawlApp(api_url=API_URL, version='v0')
|
|
24
|
+
assert "No API key provided" in str(excinfo.value)
|
|
25
|
+
|
|
26
|
+
def test_scrape_url_invalid_api_key():
|
|
27
|
+
invalid_app = FirecrawlApp(api_url=API_URL, api_key="invalid_api_key", version='v0')
|
|
28
|
+
with pytest.raises(Exception) as excinfo:
|
|
29
|
+
invalid_app.scrape_url('https://firecrawl.dev')
|
|
30
|
+
assert "Unexpected error during scrape URL: Status code 401. Unauthorized: Invalid token" in str(excinfo.value)
|
|
31
|
+
|
|
32
|
+
def test_blocklisted_url():
|
|
33
|
+
blocklisted_url = "https://facebook.com/fake-test"
|
|
34
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
35
|
+
with pytest.raises(Exception) as excinfo:
|
|
36
|
+
app.scrape_url(blocklisted_url)
|
|
37
|
+
assert "Unexpected error during scrape URL: Status code 403. Firecrawl currently does not support social media scraping due to policy restrictions. We're actively working on building support for it." in str(excinfo.value)
|
|
38
|
+
|
|
39
|
+
def test_successful_response_with_valid_preview_token():
|
|
40
|
+
app = FirecrawlApp(api_url=API_URL, api_key="this_is_just_a_preview_token", version='v0')
|
|
41
|
+
response = app.scrape_url('https://roastmywebsite.ai')
|
|
42
|
+
assert response is not None
|
|
43
|
+
assert 'content' in response
|
|
44
|
+
assert "_Roast_" in response['content']
|
|
45
|
+
|
|
46
|
+
def test_scrape_url_e2e():
|
|
47
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
48
|
+
response = app.scrape_url('https://roastmywebsite.ai')
|
|
49
|
+
print(response)
|
|
50
|
+
|
|
51
|
+
assert response is not None
|
|
52
|
+
assert 'content' in response
|
|
53
|
+
assert 'markdown' in response
|
|
54
|
+
assert 'metadata' in response
|
|
55
|
+
assert 'html' not in response
|
|
56
|
+
assert "_Roast_" in response['content']
|
|
57
|
+
|
|
58
|
+
def test_successful_response_with_valid_api_key_and_include_html():
|
|
59
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
60
|
+
response = app.scrape_url('https://roastmywebsite.ai', {'pageOptions': {'includeHtml': True}})
|
|
61
|
+
assert response is not None
|
|
62
|
+
assert 'content' in response
|
|
63
|
+
assert 'markdown' in response
|
|
64
|
+
assert 'html' in response
|
|
65
|
+
assert 'metadata' in response
|
|
66
|
+
assert "_Roast_" in response['content']
|
|
67
|
+
assert "_Roast_" in response['markdown']
|
|
68
|
+
assert "<h1" in response['html']
|
|
69
|
+
|
|
70
|
+
def test_successful_response_for_valid_scrape_with_pdf_file():
|
|
71
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
72
|
+
response = app.scrape_url('https://arxiv.org/pdf/astro-ph/9301001.pdf')
|
|
73
|
+
assert response is not None
|
|
74
|
+
assert 'content' in response
|
|
75
|
+
assert 'metadata' in response
|
|
76
|
+
assert 'We present spectrophotometric observations of the Broad Line Radio Galaxy' in response['content']
|
|
77
|
+
|
|
78
|
+
def test_successful_response_for_valid_scrape_with_pdf_file_without_explicit_extension():
|
|
79
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
80
|
+
response = app.scrape_url('https://arxiv.org/pdf/astro-ph/9301001')
|
|
81
|
+
time.sleep(6) # wait for 6 seconds
|
|
82
|
+
assert response is not None
|
|
83
|
+
assert 'content' in response
|
|
84
|
+
assert 'metadata' in response
|
|
85
|
+
assert 'We present spectrophotometric observations of the Broad Line Radio Galaxy' in response['content']
|
|
86
|
+
|
|
87
|
+
def test_crawl_url_invalid_api_key():
|
|
88
|
+
invalid_app = FirecrawlApp(api_url=API_URL, api_key="invalid_api_key", version='v0')
|
|
89
|
+
with pytest.raises(Exception) as excinfo:
|
|
90
|
+
invalid_app.crawl_url('https://firecrawl.dev')
|
|
91
|
+
assert "Unexpected error during start crawl job: Status code 401. Unauthorized: Invalid token" in str(excinfo.value)
|
|
92
|
+
|
|
93
|
+
def test_should_return_error_for_blocklisted_url():
|
|
94
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
95
|
+
blocklisted_url = "https://twitter.com/fake-test"
|
|
96
|
+
with pytest.raises(Exception) as excinfo:
|
|
97
|
+
app.crawl_url(blocklisted_url)
|
|
98
|
+
assert "Unexpected error during start crawl job: Status code 403. Firecrawl currently does not support social media scraping due to policy restrictions. We're actively working on building support for it." in str(excinfo.value)
|
|
99
|
+
|
|
100
|
+
def test_crawl_url_wait_for_completion_e2e():
|
|
101
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
102
|
+
response = app.crawl_url('https://roastmywebsite.ai', {'crawlerOptions': {'excludes': ['blog/*']}}, True)
|
|
103
|
+
assert response is not None
|
|
104
|
+
assert len(response) > 0
|
|
105
|
+
assert 'content' in response[0]
|
|
106
|
+
assert "_Roast_" in response[0]['content']
|
|
107
|
+
|
|
108
|
+
def test_crawl_url_with_idempotency_key_e2e():
|
|
109
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
110
|
+
uniqueIdempotencyKey = str(uuid4())
|
|
111
|
+
response = app.crawl_url('https://roastmywebsite.ai', {'crawlerOptions': {'excludes': ['blog/*']}}, True, 2, uniqueIdempotencyKey)
|
|
112
|
+
assert response is not None
|
|
113
|
+
assert len(response) > 0
|
|
114
|
+
assert 'content' in response[0]
|
|
115
|
+
assert "_Roast_" in response[0]['content']
|
|
116
|
+
|
|
117
|
+
with pytest.raises(Exception) as excinfo:
|
|
118
|
+
app.crawl_url('https://firecrawl.dev', {'crawlerOptions': {'excludes': ['blog/*']}}, True, 2, uniqueIdempotencyKey)
|
|
119
|
+
assert "Conflict: Failed to start crawl job due to a conflict. Idempotency key already used" in str(excinfo.value)
|
|
120
|
+
|
|
121
|
+
def test_check_crawl_status_e2e():
|
|
122
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
123
|
+
response = app.crawl_url('https://firecrawl.dev', {'crawlerOptions': {'excludes': ['blog/*']}}, False)
|
|
124
|
+
assert response is not None
|
|
125
|
+
assert 'jobId' in response
|
|
126
|
+
|
|
127
|
+
time.sleep(30) # wait for 30 seconds
|
|
128
|
+
status_response = app.check_crawl_status(response['jobId'])
|
|
129
|
+
assert status_response is not None
|
|
130
|
+
assert 'status' in status_response
|
|
131
|
+
assert status_response['status'] == 'completed'
|
|
132
|
+
assert 'data' in status_response
|
|
133
|
+
assert len(status_response['data']) > 0
|
|
134
|
+
|
|
135
|
+
def test_search_e2e():
|
|
136
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
137
|
+
response = app.search("test query")
|
|
138
|
+
assert response is not None
|
|
139
|
+
assert 'content' in response[0]
|
|
140
|
+
assert len(response) > 2
|
|
141
|
+
|
|
142
|
+
def test_search_invalid_api_key():
|
|
143
|
+
invalid_app = FirecrawlApp(api_url=API_URL, api_key="invalid_api_key", version='v0')
|
|
144
|
+
with pytest.raises(Exception) as excinfo:
|
|
145
|
+
invalid_app.search("test query")
|
|
146
|
+
assert "Unexpected error during search: Status code 401. Unauthorized: Invalid token" in str(excinfo.value)
|
|
147
|
+
|
|
148
|
+
def test_llm_extraction():
|
|
149
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY, version='v0')
|
|
150
|
+
response = app.scrape_url("https://firecrawl.dev", {
|
|
151
|
+
'extractorOptions': {
|
|
152
|
+
'mode': 'llm-extraction',
|
|
153
|
+
'extractionPrompt': "Based on the information on the page, find what the company's mission is and whether it supports SSO, and whether it is open source",
|
|
154
|
+
'extractionSchema': {
|
|
155
|
+
'type': 'object',
|
|
156
|
+
'properties': {
|
|
157
|
+
'company_mission': {'type': 'string'},
|
|
158
|
+
'supports_sso': {'type': 'boolean'},
|
|
159
|
+
'is_open_source': {'type': 'boolean'}
|
|
160
|
+
},
|
|
161
|
+
'required': ['company_mission', 'supports_sso', 'is_open_source']
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
})
|
|
165
|
+
assert response is not None
|
|
166
|
+
assert 'llm_extraction' in response
|
|
167
|
+
llm_extraction = response['llm_extraction']
|
|
168
|
+
assert 'company_mission' in llm_extraction
|
|
169
|
+
assert isinstance(llm_extraction['supports_sso'], bool)
|
|
170
|
+
assert isinstance(llm_extraction['is_open_source'], bool)
|
|
File without changes
|
|
@@ -0,0 +1,352 @@
|
|
|
1
|
+
import importlib.util
|
|
2
|
+
import pytest
|
|
3
|
+
import time
|
|
4
|
+
import os
|
|
5
|
+
from uuid import uuid4
|
|
6
|
+
from dotenv import load_dotenv
|
|
7
|
+
from datetime import datetime
|
|
8
|
+
|
|
9
|
+
load_dotenv()
|
|
10
|
+
|
|
11
|
+
API_URL = "http://127.0.0.1:3002";
|
|
12
|
+
ABSOLUTE_FIRECRAWL_PATH = "firecrawl/firecrawl.py"
|
|
13
|
+
TEST_API_KEY = os.getenv('TEST_API_KEY')
|
|
14
|
+
|
|
15
|
+
print(f"ABSOLUTE_FIRECRAWL_PATH: {ABSOLUTE_FIRECRAWL_PATH}")
|
|
16
|
+
|
|
17
|
+
spec = importlib.util.spec_from_file_location("FirecrawlApp", ABSOLUTE_FIRECRAWL_PATH)
|
|
18
|
+
firecrawl = importlib.util.module_from_spec(spec)
|
|
19
|
+
spec.loader.exec_module(firecrawl)
|
|
20
|
+
FirecrawlApp = firecrawl.FirecrawlApp
|
|
21
|
+
|
|
22
|
+
def test_no_api_key():
|
|
23
|
+
with pytest.raises(Exception) as excinfo:
|
|
24
|
+
invalid_app = FirecrawlApp(api_url=API_URL)
|
|
25
|
+
assert "No API key provided" in str(excinfo.value)
|
|
26
|
+
|
|
27
|
+
def test_scrape_url_invalid_api_key():
|
|
28
|
+
invalid_app = FirecrawlApp(api_url=API_URL, api_key="invalid_api_key")
|
|
29
|
+
with pytest.raises(Exception) as excinfo:
|
|
30
|
+
invalid_app.scrape_url('https://firecrawl.dev')
|
|
31
|
+
assert "Unauthorized: Invalid token" in str(excinfo.value)
|
|
32
|
+
|
|
33
|
+
def test_blocklisted_url():
|
|
34
|
+
blocklisted_url = "https://facebook.com/fake-test"
|
|
35
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
36
|
+
with pytest.raises(Exception) as excinfo:
|
|
37
|
+
app.scrape_url(blocklisted_url)
|
|
38
|
+
assert "URL is blocked. Firecrawl currently does not support social media scraping due to policy restrictions." in str(excinfo.value)
|
|
39
|
+
|
|
40
|
+
def test_successful_response_with_valid_preview_token():
|
|
41
|
+
app = FirecrawlApp(api_url=API_URL, api_key="this_is_just_a_preview_token")
|
|
42
|
+
response = app.scrape_url('https://roastmywebsite.ai')
|
|
43
|
+
assert response is not None
|
|
44
|
+
assert "_Roast_" in response['markdown']
|
|
45
|
+
assert "content" not in response
|
|
46
|
+
assert "html" not in response
|
|
47
|
+
assert "metadata" in response
|
|
48
|
+
assert "links" not in response
|
|
49
|
+
assert "rawHtml" not in response
|
|
50
|
+
|
|
51
|
+
def test_successful_response_for_valid_scrape():
|
|
52
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
53
|
+
response = app.scrape_url('https://roastmywebsite.ai')
|
|
54
|
+
assert response is not None
|
|
55
|
+
assert 'markdown' in response
|
|
56
|
+
assert "_Roast_" in response['markdown']
|
|
57
|
+
assert 'metadata' in response
|
|
58
|
+
assert 'content' not in response
|
|
59
|
+
assert 'html' not in response
|
|
60
|
+
assert 'rawHtml' not in response
|
|
61
|
+
assert 'screenshot' not in response
|
|
62
|
+
assert 'links' not in response
|
|
63
|
+
|
|
64
|
+
def test_successful_response_with_valid_api_key_and_options():
|
|
65
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
66
|
+
params = {
|
|
67
|
+
'formats': ['markdown', 'html', 'rawHtml', 'screenshot', 'links'],
|
|
68
|
+
'headers': {'x-key': 'test'},
|
|
69
|
+
'includeTags': ['h1'],
|
|
70
|
+
'excludeTags': ['h2'],
|
|
71
|
+
'onlyMainContent': True,
|
|
72
|
+
'timeout': 30000,
|
|
73
|
+
'waitFor': 1000
|
|
74
|
+
}
|
|
75
|
+
response = app.scrape_url('https://roastmywebsite.ai', params)
|
|
76
|
+
assert response is not None
|
|
77
|
+
assert 'content' not in response
|
|
78
|
+
assert 'markdown' in response
|
|
79
|
+
assert 'html' in response
|
|
80
|
+
assert 'rawHtml' in response
|
|
81
|
+
assert 'screenshot' in response
|
|
82
|
+
assert 'links' in response
|
|
83
|
+
assert "_Roast_" in response['markdown']
|
|
84
|
+
assert "<h1" in response['html']
|
|
85
|
+
assert "<h1" in response['rawHtml']
|
|
86
|
+
assert "https://" in response['screenshot']
|
|
87
|
+
assert len(response['links']) > 0
|
|
88
|
+
assert "https://" in response['links'][0]
|
|
89
|
+
assert 'metadata' in response
|
|
90
|
+
assert 'title' in response['metadata']
|
|
91
|
+
assert 'description' in response['metadata']
|
|
92
|
+
assert 'keywords' in response['metadata']
|
|
93
|
+
assert 'robots' in response['metadata']
|
|
94
|
+
assert 'ogTitle' in response['metadata']
|
|
95
|
+
assert 'ogDescription' in response['metadata']
|
|
96
|
+
assert 'ogUrl' in response['metadata']
|
|
97
|
+
assert 'ogImage' in response['metadata']
|
|
98
|
+
assert 'ogLocaleAlternate' in response['metadata']
|
|
99
|
+
assert 'ogSiteName' in response['metadata']
|
|
100
|
+
assert 'sourceURL' in response['metadata']
|
|
101
|
+
assert 'statusCode' in response['metadata']
|
|
102
|
+
assert 'pageStatusCode' not in response['metadata']
|
|
103
|
+
assert 'pageError' not in response['metadata']
|
|
104
|
+
assert 'error' not in response['metadata']
|
|
105
|
+
assert response['metadata']['title'] == "Roast My Website"
|
|
106
|
+
assert response['metadata']['description'] == "Welcome to Roast My Website, the ultimate tool for putting your website through the wringer! This repository harnesses the power of Firecrawl to scrape and capture screenshots of websites, and then unleashes the latest LLM vision models to mercilessly roast them. 🌶️"
|
|
107
|
+
assert response['metadata']['keywords'] == "Roast My Website,Roast,Website,GitHub,Firecrawl"
|
|
108
|
+
assert response['metadata']['robots'] == "follow, index"
|
|
109
|
+
assert response['metadata']['ogTitle'] == "Roast My Website"
|
|
110
|
+
assert response['metadata']['ogDescription'] == "Welcome to Roast My Website, the ultimate tool for putting your website through the wringer! This repository harnesses the power of Firecrawl to scrape and capture screenshots of websites, and then unleashes the latest LLM vision models to mercilessly roast them. 🌶️"
|
|
111
|
+
assert response['metadata']['ogUrl'] == "https://www.roastmywebsite.ai"
|
|
112
|
+
assert response['metadata']['ogImage'] == "https://www.roastmywebsite.ai/og.png"
|
|
113
|
+
assert response['metadata']['ogLocaleAlternate'] == []
|
|
114
|
+
assert response['metadata']['ogSiteName'] == "Roast My Website"
|
|
115
|
+
assert response['metadata']['sourceURL'] == "https://roastmywebsite.ai"
|
|
116
|
+
assert response['metadata']['statusCode'] == 200
|
|
117
|
+
|
|
118
|
+
def test_successful_response_for_valid_scrape_with_pdf_file():
|
|
119
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
120
|
+
response = app.scrape_url('https://arxiv.org/pdf/astro-ph/9301001.pdf')
|
|
121
|
+
assert response is not None
|
|
122
|
+
assert 'content' not in response
|
|
123
|
+
assert 'metadata' in response
|
|
124
|
+
assert 'We present spectrophotometric observations of the Broad Line Radio Galaxy' in response['markdown']
|
|
125
|
+
|
|
126
|
+
def test_successful_response_for_valid_scrape_with_pdf_file_without_explicit_extension():
|
|
127
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
128
|
+
response = app.scrape_url('https://arxiv.org/pdf/astro-ph/9301001')
|
|
129
|
+
time.sleep(1) # wait for 1 second
|
|
130
|
+
assert response is not None
|
|
131
|
+
assert 'We present spectrophotometric observations of the Broad Line Radio Galaxy' in response['markdown']
|
|
132
|
+
|
|
133
|
+
def test_crawl_url_invalid_api_key():
|
|
134
|
+
invalid_app = FirecrawlApp(api_url=API_URL, api_key="invalid_api_key")
|
|
135
|
+
with pytest.raises(Exception) as excinfo:
|
|
136
|
+
invalid_app.crawl_url('https://firecrawl.dev')
|
|
137
|
+
assert "Unauthorized: Invalid token" in str(excinfo.value)
|
|
138
|
+
|
|
139
|
+
def test_should_return_error_for_blocklisted_url():
|
|
140
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
141
|
+
blocklisted_url = "https://twitter.com/fake-test"
|
|
142
|
+
with pytest.raises(Exception) as excinfo:
|
|
143
|
+
app.crawl_url(blocklisted_url)
|
|
144
|
+
assert "URL is blocked. Firecrawl currently does not support social media scraping due to policy restrictions." in str(excinfo.value)
|
|
145
|
+
|
|
146
|
+
def test_crawl_url_wait_for_completion_e2e():
|
|
147
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
148
|
+
response = app.crawl_url('https://roastmywebsite.ai', {'excludePaths': ['blog/*']}, True, 30)
|
|
149
|
+
assert response is not None
|
|
150
|
+
assert 'total' in response
|
|
151
|
+
assert response['total'] > 0
|
|
152
|
+
assert 'creditsUsed' in response
|
|
153
|
+
assert response['creditsUsed'] > 0
|
|
154
|
+
assert 'expiresAt' in response
|
|
155
|
+
assert datetime.strptime(response['expiresAt'], '%Y-%m-%dT%H:%M:%S.%fZ') > datetime.now()
|
|
156
|
+
assert 'status' in response
|
|
157
|
+
assert response['status'] == 'completed'
|
|
158
|
+
assert 'next' not in response
|
|
159
|
+
assert len(response['data']) > 0
|
|
160
|
+
assert 'markdown' in response['data'][0]
|
|
161
|
+
assert "_Roast_" in response['data'][0]['markdown']
|
|
162
|
+
assert 'content' not in response['data'][0]
|
|
163
|
+
assert 'html' not in response['data'][0]
|
|
164
|
+
assert 'rawHtml' not in response['data'][0]
|
|
165
|
+
assert 'screenshot' not in response['data'][0]
|
|
166
|
+
assert 'links' not in response['data'][0]
|
|
167
|
+
assert 'metadata' in response['data'][0]
|
|
168
|
+
assert 'title' in response['data'][0]['metadata']
|
|
169
|
+
assert 'description' in response['data'][0]['metadata']
|
|
170
|
+
assert 'language' in response['data'][0]['metadata']
|
|
171
|
+
assert 'sourceURL' in response['data'][0]['metadata']
|
|
172
|
+
assert 'statusCode' in response['data'][0]['metadata']
|
|
173
|
+
assert 'error' not in response['data'][0]['metadata']
|
|
174
|
+
|
|
175
|
+
def test_crawl_url_with_options_and_wait_for_completion():
|
|
176
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
177
|
+
response = app.crawl_url('https://roastmywebsite.ai', {
|
|
178
|
+
'excludePaths': ['blog/*'],
|
|
179
|
+
'includePaths': ['/'],
|
|
180
|
+
'maxDepth': 2,
|
|
181
|
+
'ignoreSitemap': True,
|
|
182
|
+
'limit': 10,
|
|
183
|
+
'allowBackwardLinks': True,
|
|
184
|
+
'allowExternalLinks': True,
|
|
185
|
+
'scrapeOptions': {
|
|
186
|
+
'formats': ['markdown', 'html', 'rawHtml', 'screenshot', 'links'],
|
|
187
|
+
'headers': {"x-key": "test"},
|
|
188
|
+
'includeTags': ['h1'],
|
|
189
|
+
'excludeTags': ['h2'],
|
|
190
|
+
'onlyMainContent': True,
|
|
191
|
+
'waitFor': 1000
|
|
192
|
+
}
|
|
193
|
+
}, True, 30)
|
|
194
|
+
assert response is not None
|
|
195
|
+
assert 'total' in response
|
|
196
|
+
assert response['total'] > 0
|
|
197
|
+
assert 'creditsUsed' in response
|
|
198
|
+
assert response['creditsUsed'] > 0
|
|
199
|
+
assert 'expiresAt' in response
|
|
200
|
+
assert datetime.strptime(response['expiresAt'], '%Y-%m-%dT%H:%M:%S.%fZ') > datetime.now()
|
|
201
|
+
assert 'status' in response
|
|
202
|
+
assert response['status'] == 'completed'
|
|
203
|
+
assert 'next' not in response
|
|
204
|
+
assert len(response['data']) > 0
|
|
205
|
+
assert 'markdown' in response['data'][0]
|
|
206
|
+
assert "_Roast_" in response['data'][0]['markdown']
|
|
207
|
+
assert 'content' not in response['data'][0]
|
|
208
|
+
assert 'html' in response['data'][0]
|
|
209
|
+
assert "<h1" in response['data'][0]['html']
|
|
210
|
+
assert 'rawHtml' in response['data'][0]
|
|
211
|
+
assert "<h1" in response['data'][0]['rawHtml']
|
|
212
|
+
assert 'screenshot' in response['data'][0]
|
|
213
|
+
assert "https://" in response['data'][0]['screenshot']
|
|
214
|
+
assert 'links' in response['data'][0]
|
|
215
|
+
assert len(response['data'][0]['links']) > 0
|
|
216
|
+
assert 'metadata' in response['data'][0]
|
|
217
|
+
assert 'title' in response['data'][0]['metadata']
|
|
218
|
+
assert 'description' in response['data'][0]['metadata']
|
|
219
|
+
assert 'language' in response['data'][0]['metadata']
|
|
220
|
+
assert 'sourceURL' in response['data'][0]['metadata']
|
|
221
|
+
assert 'statusCode' in response['data'][0]['metadata']
|
|
222
|
+
assert 'error' not in response['data'][0]['metadata']
|
|
223
|
+
|
|
224
|
+
def test_crawl_url_with_idempotency_key_e2e():
|
|
225
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
226
|
+
uniqueIdempotencyKey = str(uuid4())
|
|
227
|
+
response = app.crawl_url('https://roastmywebsite.ai', {'excludePaths': ['blog/*']}, False, 2, uniqueIdempotencyKey)
|
|
228
|
+
assert response is not None
|
|
229
|
+
assert 'id' in response
|
|
230
|
+
|
|
231
|
+
with pytest.raises(Exception) as excinfo:
|
|
232
|
+
app.crawl_url('https://firecrawl.dev', {'excludePaths': ['blog/*']}, True, 2, uniqueIdempotencyKey)
|
|
233
|
+
assert "Idempotency key already used" in str(excinfo.value)
|
|
234
|
+
|
|
235
|
+
def test_check_crawl_status_e2e():
|
|
236
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
237
|
+
response = app.crawl_url('https://firecrawl.dev', {'scrapeOptions': {'formats': ['markdown', 'html', 'rawHtml', 'screenshot', 'links']}}, False)
|
|
238
|
+
assert response is not None
|
|
239
|
+
assert 'id' in response
|
|
240
|
+
|
|
241
|
+
max_checks = 15
|
|
242
|
+
checks = 0
|
|
243
|
+
status_response = app.check_crawl_status(response['id'])
|
|
244
|
+
|
|
245
|
+
while status_response['status'] == 'scraping' and checks < max_checks:
|
|
246
|
+
time.sleep(1) # wait for 1 second
|
|
247
|
+
assert 'partial_data' not in status_response
|
|
248
|
+
assert 'current' not in status_response
|
|
249
|
+
assert 'data' in status_response
|
|
250
|
+
assert 'total' in status_response
|
|
251
|
+
assert 'creditsUsed' in status_response
|
|
252
|
+
assert 'expiresAt' in status_response
|
|
253
|
+
assert 'status' in status_response
|
|
254
|
+
assert 'next' in status_response
|
|
255
|
+
assert status_response['total'] > 0
|
|
256
|
+
assert status_response['creditsUsed'] > 0
|
|
257
|
+
assert datetime.strptime(status_response['expiresAt'], '%Y-%m-%dT%H:%M:%S.%fZ') > datetime.now()
|
|
258
|
+
assert status_response['status'] == 'scraping'
|
|
259
|
+
assert '/v1/crawl/' in status_response['next']
|
|
260
|
+
status_response = app.check_crawl_status(response['id'])
|
|
261
|
+
checks += 1
|
|
262
|
+
|
|
263
|
+
assert status_response is not None
|
|
264
|
+
assert 'total' in status_response
|
|
265
|
+
assert status_response['total'] > 0
|
|
266
|
+
assert 'creditsUsed' in status_response
|
|
267
|
+
assert status_response['creditsUsed'] > 0
|
|
268
|
+
assert 'expiresAt' in status_response
|
|
269
|
+
assert datetime.strptime(status_response['expiresAt'], '%Y-%m-%dT%H:%M:%S.%fZ') > datetime.now()
|
|
270
|
+
assert 'status' in status_response
|
|
271
|
+
assert status_response['status'] == 'completed'
|
|
272
|
+
assert len(status_response['data']) > 0
|
|
273
|
+
assert 'markdown' in status_response['data'][0]
|
|
274
|
+
assert len(status_response['data'][0]['markdown']) > 10
|
|
275
|
+
assert 'content' not in status_response['data'][0]
|
|
276
|
+
assert 'html' in status_response['data'][0]
|
|
277
|
+
assert "<div" in status_response['data'][0]['html']
|
|
278
|
+
assert 'rawHtml' in status_response['data'][0]
|
|
279
|
+
assert "<div" in status_response['data'][0]['rawHtml']
|
|
280
|
+
assert 'screenshot' in status_response['data'][0]
|
|
281
|
+
assert "https://" in status_response['data'][0]['screenshot']
|
|
282
|
+
assert 'links' in status_response['data'][0]
|
|
283
|
+
assert status_response['data'][0]['links'] is not None
|
|
284
|
+
assert len(status_response['data'][0]['links']) > 0
|
|
285
|
+
assert 'metadata' in status_response['data'][0]
|
|
286
|
+
assert 'title' in status_response['data'][0]['metadata']
|
|
287
|
+
assert 'description' in status_response['data'][0]['metadata']
|
|
288
|
+
assert 'language' in status_response['data'][0]['metadata']
|
|
289
|
+
assert 'sourceURL' in status_response['data'][0]['metadata']
|
|
290
|
+
assert 'statusCode' in status_response['data'][0]['metadata']
|
|
291
|
+
assert 'error' not in status_response['data'][0]['metadata']
|
|
292
|
+
|
|
293
|
+
def test_invalid_api_key_on_map():
|
|
294
|
+
invalid_app = FirecrawlApp(api_key="invalid_api_key", api_url=API_URL)
|
|
295
|
+
with pytest.raises(Exception) as excinfo:
|
|
296
|
+
invalid_app.map_url('https://roastmywebsite.ai')
|
|
297
|
+
assert "Unauthorized: Invalid token" in str(excinfo.value)
|
|
298
|
+
|
|
299
|
+
def test_blocklisted_url_on_map():
|
|
300
|
+
app = FirecrawlApp(api_key=TEST_API_KEY, api_url=API_URL)
|
|
301
|
+
blocklisted_url = "https://facebook.com/fake-test"
|
|
302
|
+
with pytest.raises(Exception) as excinfo:
|
|
303
|
+
app.map_url(blocklisted_url)
|
|
304
|
+
assert "URL is blocked. Firecrawl currently does not support social media scraping due to policy restrictions." in str(excinfo.value)
|
|
305
|
+
|
|
306
|
+
def test_successful_response_with_valid_preview_token_on_map():
|
|
307
|
+
app = FirecrawlApp(api_key="this_is_just_a_preview_token", api_url=API_URL)
|
|
308
|
+
response = app.map_url('https://roastmywebsite.ai')
|
|
309
|
+
assert response is not None
|
|
310
|
+
assert len(response) > 0
|
|
311
|
+
|
|
312
|
+
def test_successful_response_for_valid_map():
|
|
313
|
+
app = FirecrawlApp(api_key=TEST_API_KEY, api_url=API_URL)
|
|
314
|
+
response = app.map_url('https://roastmywebsite.ai')
|
|
315
|
+
assert response is not None
|
|
316
|
+
assert len(response) > 0
|
|
317
|
+
assert any("https://" in link for link in response)
|
|
318
|
+
filtered_links = [link for link in response if "roastmywebsite.ai" in link]
|
|
319
|
+
assert len(filtered_links) > 0
|
|
320
|
+
|
|
321
|
+
def test_search_e2e():
|
|
322
|
+
app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
323
|
+
with pytest.raises(NotImplementedError) as excinfo:
|
|
324
|
+
app.search("test query")
|
|
325
|
+
assert "Search is not supported in v1" in str(excinfo.value)
|
|
326
|
+
|
|
327
|
+
# def test_llm_extraction():
|
|
328
|
+
# app = FirecrawlApp(api_url=API_URL, api_key=TEST_API_KEY)
|
|
329
|
+
# response = app.scrape_url("https://mendable.ai", {
|
|
330
|
+
# 'extractorOptions': {
|
|
331
|
+
# 'mode': 'llm-extraction',
|
|
332
|
+
# 'extractionPrompt': "Based on the information on the page, find what the company's mission is and whether it supports SSO, and whether it is open source",
|
|
333
|
+
# 'extractionSchema': {
|
|
334
|
+
# 'type': 'object',
|
|
335
|
+
# 'properties': {
|
|
336
|
+
# 'company_mission': {'type': 'string'},
|
|
337
|
+
# 'supports_sso': {'type': 'boolean'},
|
|
338
|
+
# 'is_open_source': {'type': 'boolean'}
|
|
339
|
+
# },
|
|
340
|
+
# 'required': ['company_mission', 'supports_sso', 'is_open_source']
|
|
341
|
+
# }
|
|
342
|
+
# }
|
|
343
|
+
# })
|
|
344
|
+
# assert response is not None
|
|
345
|
+
# assert 'llm_extraction' in response
|
|
346
|
+
# llm_extraction = response['llm_extraction']
|
|
347
|
+
# assert 'company_mission' in llm_extraction
|
|
348
|
+
# assert isinstance(llm_extraction['supports_sso'], bool)
|
|
349
|
+
# assert isinstance(llm_extraction['is_open_source'], bool)
|
|
350
|
+
|
|
351
|
+
|
|
352
|
+
|
|
@@ -189,17 +189,38 @@ class FirecrawlApp:
|
|
|
189
189
|
headers = self._prepare_headers()
|
|
190
190
|
response = self._get_request(f'{self.api_url}{endpoint}', headers)
|
|
191
191
|
if response.status_code == 200:
|
|
192
|
-
|
|
192
|
+
status_data = response.json()
|
|
193
|
+
if status_data['status'] == 'completed':
|
|
194
|
+
if 'data' in status_data:
|
|
195
|
+
data = status_data['data']
|
|
196
|
+
while 'next' in status_data:
|
|
197
|
+
next_url = status_data.get('next')
|
|
198
|
+
if not next_url:
|
|
199
|
+
logger.warning("Expected 'next' URL is missing.")
|
|
200
|
+
break
|
|
201
|
+
try:
|
|
202
|
+
status_response = self._get_request(next_url, headers)
|
|
203
|
+
if status_response.status_code != 200:
|
|
204
|
+
logger.error(f"Failed to fetch next page: {status_response.status_code}")
|
|
205
|
+
break
|
|
206
|
+
status_data = status_response.json()
|
|
207
|
+
data.extend(status_data.get('data', []))
|
|
208
|
+
except Exception as e:
|
|
209
|
+
logger.error(f"Error during pagination request: {e}")
|
|
210
|
+
break
|
|
211
|
+
status_data.pop('next', None)
|
|
212
|
+
status_data['data'] = data
|
|
213
|
+
|
|
193
214
|
return {
|
|
194
215
|
'success': True,
|
|
195
|
-
'status':
|
|
196
|
-
'total':
|
|
197
|
-
'completed':
|
|
198
|
-
'creditsUsed':
|
|
199
|
-
'expiresAt':
|
|
200
|
-
'
|
|
201
|
-
'
|
|
202
|
-
'
|
|
216
|
+
'status': status_data.get('status'),
|
|
217
|
+
'total': status_data.get('total'),
|
|
218
|
+
'completed': status_data.get('completed'),
|
|
219
|
+
'creditsUsed': status_data.get('creditsUsed'),
|
|
220
|
+
'expiresAt': status_data.get('expiresAt'),
|
|
221
|
+
'data': status_data.get('data'),
|
|
222
|
+
'error': status_data.get('error'),
|
|
223
|
+
'next': status_data.get('next', None)
|
|
203
224
|
}
|
|
204
225
|
else:
|
|
205
226
|
self._handle_error(response, 'check crawl status')
|
|
@@ -377,17 +398,38 @@ class FirecrawlApp:
|
|
|
377
398
|
headers = self._prepare_headers()
|
|
378
399
|
response = self._get_request(f'{self.api_url}{endpoint}', headers)
|
|
379
400
|
if response.status_code == 200:
|
|
380
|
-
|
|
401
|
+
status_data = response.json()
|
|
402
|
+
if status_data['status'] == 'completed':
|
|
403
|
+
if 'data' in status_data:
|
|
404
|
+
data = status_data['data']
|
|
405
|
+
while 'next' in status_data:
|
|
406
|
+
next_url = status_data.get('next')
|
|
407
|
+
if not next_url:
|
|
408
|
+
logger.warning("Expected 'next' URL is missing.")
|
|
409
|
+
break
|
|
410
|
+
try:
|
|
411
|
+
status_response = self._get_request(next_url, headers)
|
|
412
|
+
if status_response.status_code != 200:
|
|
413
|
+
logger.error(f"Failed to fetch next page: {status_response.status_code}")
|
|
414
|
+
break
|
|
415
|
+
status_data = status_response.json()
|
|
416
|
+
data.extend(status_data.get('data', []))
|
|
417
|
+
except Exception as e:
|
|
418
|
+
logger.error(f"Error during pagination request: {e}")
|
|
419
|
+
break
|
|
420
|
+
status_data.pop('next', None)
|
|
421
|
+
status_data['data'] = data
|
|
422
|
+
|
|
381
423
|
return {
|
|
382
424
|
'success': True,
|
|
383
|
-
'status':
|
|
384
|
-
'total':
|
|
385
|
-
'completed':
|
|
386
|
-
'creditsUsed':
|
|
387
|
-
'expiresAt':
|
|
388
|
-
'
|
|
389
|
-
'
|
|
390
|
-
'
|
|
425
|
+
'status': status_data.get('status'),
|
|
426
|
+
'total': status_data.get('total'),
|
|
427
|
+
'completed': status_data.get('completed'),
|
|
428
|
+
'creditsUsed': status_data.get('creditsUsed'),
|
|
429
|
+
'expiresAt': status_data.get('expiresAt'),
|
|
430
|
+
'data': status_data.get('data'),
|
|
431
|
+
'error': status_data.get('error'),
|
|
432
|
+
'next': status_data.get('next', None)
|
|
391
433
|
}
|
|
392
434
|
else:
|
|
393
435
|
self._handle_error(response, 'check batch scrape status')
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: firecrawl
|
|
3
|
-
Version: 1.
|
|
3
|
+
Version: 1.5.0
|
|
4
4
|
Summary: Python SDK for Firecrawl API
|
|
5
5
|
Home-page: https://github.com/mendableai/firecrawl
|
|
6
6
|
Author: Mendable.ai
|
|
@@ -210,11 +210,11 @@ print(batch_scrape_result)
|
|
|
210
210
|
|
|
211
211
|
### Checking batch scrape status
|
|
212
212
|
|
|
213
|
-
To check the status of an asynchronous batch scrape job, use the `
|
|
213
|
+
To check the status of an asynchronous batch scrape job, use the `check_batch_scrape_status` method. It takes the job ID as a parameter and returns the current status of the batch scrape job.
|
|
214
214
|
|
|
215
215
|
```python
|
|
216
216
|
id = batch_scrape_result['id']
|
|
217
|
-
status = app.
|
|
217
|
+
status = app.check_batch_scrape_status(id)
|
|
218
218
|
```
|
|
219
219
|
|
|
220
220
|
### Batch scrape with WebSockets
|
|
@@ -8,4 +8,8 @@ firecrawl.egg-info/PKG-INFO
|
|
|
8
8
|
firecrawl.egg-info/SOURCES.txt
|
|
9
9
|
firecrawl.egg-info/dependency_links.txt
|
|
10
10
|
firecrawl.egg-info/requires.txt
|
|
11
|
-
firecrawl.egg-info/top_level.txt
|
|
11
|
+
firecrawl.egg-info/top_level.txt
|
|
12
|
+
firecrawl/__tests__/e2e_withAuth/__init__.py
|
|
13
|
+
firecrawl/__tests__/e2e_withAuth/test.py
|
|
14
|
+
firecrawl/__tests__/v1/e2e_withAuth/__init__.py
|
|
15
|
+
firecrawl/__tests__/v1/e2e_withAuth/test.py
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
firecrawl
|
|
@@ -1,57 +0,0 @@
|
|
|
1
|
-
"""
|
|
2
|
-
This is the Firecrawl package.
|
|
3
|
-
|
|
4
|
-
This package provides a Python SDK for interacting with the Firecrawl API.
|
|
5
|
-
It includes methods to scrape URLs, perform searches, initiate and monitor crawl jobs,
|
|
6
|
-
and check the status of these jobs.
|
|
7
|
-
|
|
8
|
-
For more information visit https://github.com/firecrawl/
|
|
9
|
-
"""
|
|
10
|
-
|
|
11
|
-
import logging
|
|
12
|
-
import os
|
|
13
|
-
|
|
14
|
-
from .firecrawl import FirecrawlApp
|
|
15
|
-
|
|
16
|
-
__version__ = "1.4.0"
|
|
17
|
-
|
|
18
|
-
# Define the logger for the Firecrawl project
|
|
19
|
-
logger: logging.Logger = logging.getLogger("firecrawl")
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
def _basic_config() -> None:
|
|
23
|
-
"""Set up basic configuration for logging with a specific format and date format."""
|
|
24
|
-
try:
|
|
25
|
-
logging.basicConfig(
|
|
26
|
-
format="[%(asctime)s - %(name)s:%(lineno)d - %(levelname)s] %(message)s",
|
|
27
|
-
datefmt="%Y-%m-%d %H:%M:%S",
|
|
28
|
-
)
|
|
29
|
-
except Exception as e:
|
|
30
|
-
logger.error("Failed to configure logging: %s", e)
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
def setup_logging() -> None:
|
|
34
|
-
"""Set up logging based on the FIRECRAWL_LOGGING_LEVEL environment variable."""
|
|
35
|
-
env = os.environ.get(
|
|
36
|
-
"FIRECRAWL_LOGGING_LEVEL", "INFO"
|
|
37
|
-
).upper() # Default to 'INFO' level
|
|
38
|
-
_basic_config()
|
|
39
|
-
|
|
40
|
-
if env == "DEBUG":
|
|
41
|
-
logger.setLevel(logging.DEBUG)
|
|
42
|
-
elif env == "INFO":
|
|
43
|
-
logger.setLevel(logging.INFO)
|
|
44
|
-
elif env == "WARNING":
|
|
45
|
-
logger.setLevel(logging.WARNING)
|
|
46
|
-
elif env == "ERROR":
|
|
47
|
-
logger.setLevel(logging.ERROR)
|
|
48
|
-
elif env == "CRITICAL":
|
|
49
|
-
logger.setLevel(logging.CRITICAL)
|
|
50
|
-
else:
|
|
51
|
-
logger.setLevel(logging.INFO)
|
|
52
|
-
logger.warning("Unknown logging level: %s, defaulting to INFO", env)
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
# Initialize logging configuration when the module is imported
|
|
56
|
-
setup_logging()
|
|
57
|
-
logger.debug("Debugging logger setup")
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|