# Copyright 2010-2025 Google LLC # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # ---------- Python support ---------- .PHONY: help_python # Generate list of Python targets with descriptions. help_python: @echo Use one of the following Python targets: ifeq ($(PLATFORM),WIN64) @$(GREP) "^.PHONY: .* #" $(CURDIR)/makefiles/Makefile.python.mk | $(SED) "s/\.PHONY: \(.*\) # \(.*\)/\1\t\2/" @echo off & echo( else @$(GREP) "^.PHONY: .* #" $(CURDIR)/makefiles/Makefile.python.mk | $(SED) "s/\.PHONY: \(.*\) # \(.*\)/\1\t\2/" | expand -t24 @echo endif # Main targets. .PHONY: detect_python # Show variables used to build Python OR-Tools. .PHONY: python # Build Python OR-Tools. .PHONY: check_python # Quick check only running Python OR-Tools samples. .PHONY: test_python # Run all Python OR-Tools test targets. .PHONY: archive_python # Generate Python OR-Tools archive. .PHONY: clean_python # Clean Python output from previous build. # deprecated .PHONY: package_python # Create Python ortools Wheel package. .PHONY: test_package_python # Test Python "ortools" Wheel package .PHONY: install_python # Install Python OR-Tools on the host system ifeq ($(HAS_PYTHON),OFF) python: $(warning Either Python support was turned off, or the python3 binary was not found.) check_python: python test_python: python package_python: python else # HAS_PYTHON=ON PYTHON_BUILD_PATH := $(BUILD_DIR)$Spython # OR Tools unique library. python: $(MAKE) third_party BUILD_PYTHON=ON BUILD_VENV=ON cmake --build $(BUILD_DIR) --target install --config $(BUILD_TYPE) -j $(JOBS) -v ifeq ($(PLATFORM),WIN64) PYTHON_EXECUTABLE := $(PYTHON_BUILD_PATH)$Svenv$SScripts$Spython else PYTHON_EXECUTABLE := $(PYTHON_BUILD_PATH)$Svenv$Sbin$Spython endif ####################### ## Python SOURCE ## ####################### ifeq ($(SOURCE_SUFFIX),.py) # Those rules will be used if SOURCE contains a .py file .PHONY: run # Run a Python program. run: "$(PYTHON_EXECUTABLE)" $(SOURCE_PATH) $(ARGS) endif ##################################### ## Python Samples/Examples/Tests ## ##################################### # Samples define python-sample-target = rpy_%: \ python \ $(SRC_DIR)/ortools/$1/samples/%.py \ FORCE "$(PYTHON_EXECUTABLE)" ortools$S$1$Ssamples$S$$*.py $(ARGS) endef PYTHON_SAMPLES := algorithms graph constraint_solver linear_solver math_opt pdlp routing sat set_cover $(foreach sample,$(PYTHON_SAMPLES),$(eval $(call python-sample-target,$(sample)))) # Examples define python-example-target = rpy_%: \ python \ $(SRC_DIR)/examples/$1/%.py \ FORCE "$(PYTHON_EXECUTABLE)" examples$S$1$S$$*.py $(ARGS) endef PYTHON_EXAMPLES := contrib python $(foreach example,$(PYTHON_EXAMPLES),$(eval $(call python-example-target,$(example)))) # Tests define python-test-target = rpy_%: \ python \ $(SRC_DIR)/ortools/$1/python/%.py \ FORCE "$(PYTHON_EXECUTABLE)" ortools$S$1$Spython$S$$*.py $(ARGS) endef PYTHON_TESTS := init algorithms graph constraint_solver linear_solver math_opt routing sat util $(foreach test,$(PYTHON_TESTS),$(eval $(call python-test-target,$(test)))) #################### ## Test targets ## #################### .PHONY: test_python_algorithms_samples # Run all Python Algorithms Samples (located in ortools/algorithms/samples) test_python_algorithms_samples: \ rpy_knapsack \ rpy_simple_knapsack_program .PHONY: test_python_constraint_solver_samples # Run all Python CP Samples (located in ortools/constraint_solver/samples) test_python_constraint_solver_samples: \ rpy_nqueens_cp \ rpy_simple_cp_program \ rpy_simple_routing_program \ rpy_tsp \ rpy_tsp_cities \ rpy_tsp_circuit_board \ rpy_tsp_distance_matrix \ rpy_vrp \ rpy_vrp_breaks \ rpy_vrp_breaks_from_start \ rpy_vrp_capacity \ rpy_vrp_drop_nodes \ rpy_vrp_global_span \ rpy_vrp_initial_routes \ rpy_vrp_nodes_indices \ rpy_vrp_pickup_delivery \ rpy_vrp_pickup_delivery_fifo \ rpy_vrp_pickup_delivery_lifo \ rpy_vrp_resources \ rpy_vrp_solution_callback \ rpy_vrp_starts_ends \ rpy_vrp_time_windows \ rpy_vrp_tokens \ rpy_vrp_with_time_limit \ rpy_vrpgs \ rpy_cvrp \ rpy_cvrp_reload \ rpy_cvrptw \ rpy_cvrptw_break .PHONY: test_python_graph_samples # Run all Python Graph Samples (located in ortools/graph/samples) test_python_graph_samples: \ rpy_assignment_linear_sum_assignment \ rpy_assignment_min_flow \ rpy_balance_min_flow \ rpy_simple_max_flow_program \ rpy_simple_min_cost_flow_program .PHONY: test_python_linear_solver_samples # Run all Python LP Samples (located in ortools/linear_solver/samples) test_python_linear_solver_samples: \ rpy_assignment_mip \ rpy_basic_example \ rpy_bin_packing_mip \ rpy_integer_programming_example \ rpy_linear_programming_example \ rpy_mip_var_array \ rpy_multiple_knapsack_mip \ rpy_simple_lp_program \ rpy_simple_mip_program \ rpy_stigler_diet .PHONY: test_python_model_builder_samples # Run all Python Model Builder Samples (located in ortools/model_builder/samples) test_python_model_builder_samples: \ rpy_assignment_mb \ rpy_bin_packing_mb \ rpy_simple_lp_program_mb \ rpy_simple_mip_program_mb .PHONY: test_python_sat_samples # Run all Python Sat Samples (located in ortools/sat/samples) test_python_sat_samples: \ rpy_assignment_groups_sat \ rpy_assignment_sat \ rpy_assignment_task_sizes_sat \ rpy_assignment_teams_sat \ rpy_assumptions_sample_sat \ rpy_binpacking_problem_sat \ rpy_bool_or_sample_sat \ rpy_boolean_product_sample_sat \ rpy_channeling_sample_sat \ rpy_copy_model_sample_sat \ rpy_cp_is_fun_sat \ rpy_cp_sat_example \ rpy_earliness_tardiness_cost_sample_sat \ rpy_interval_sample_sat \ rpy_literal_sample_sat \ rpy_minimal_jobshop_sat \ rpy_multiple_knapsack_sat \ rpy_no_overlap_sample_sat \ rpy_non_linear_sat \ rpy_nqueens_sat \ rpy_nurses_sat \ rpy_optional_interval_sample_sat \ rpy_overlapping_intervals_sample_sat \ rpy_rabbits_and_pheasants_sat \ rpy_ranking_sample_sat \ rpy_reified_sample_sat \ rpy_schedule_requests_sat \ rpy_scheduling_with_calendar_sample_sat \ rpy_search_for_all_solutions_sample_sat \ rpy_simple_sat_program \ rpy_solution_hinting_sample_sat \ rpy_solve_and_print_intermediate_solutions_sample_sat \ rpy_solve_with_time_limit_sample_sat \ rpy_step_function_sample_sat \ rpy_stop_after_n_solutions_sample_sat check_python: \ test_python_algorithms_samples \ test_python_constraint_solver_samples \ test_python_graph_samples \ test_python_linear_solver_samples \ test_python_model_builder_samples \ test_python_sat_samples \ # rpy_rabbits_pheasants_cp \ # rpy_cryptarithmetic_cp \ # rpy_cryptarithmetic_sat \ # rpy_nqueens_cp \ # rpy_nqueens_sat \ # rpy_integer_programming \ # rpy_max_flow \ # rpy_min_cost_flow \ # rpy_assignment \ # rpy_nurses_cp \ # rpy_job_shop_cp \ .PHONY: test_python_tests # Run all Python Tests (located in examples/tests) test_python_tests: \ rpy_init_test \ rpy_lp_api_test \ rpy_lp_test \ rpy_model_builder_test \ rpy_pywraplp_test \ rpy_cp_api_test \ rpy_routing_api_test \ rpy_pywrapcp_test \ rpy_pywraprouting_test \ rpy_cp_model_test \ rpy_sorted_interval_list_test .PHONY: test_python_contrib # Run all Python Contrib (located in examples/python and examples/contrib) test_python_contrib: \ rpy_3_jugs_mip \ rpy_3_jugs_regular \ rpy_alldifferent_except_0 \ rpy_all_interval \ rpy_alphametic \ rpy_a_round_of_golf \ rpy_assignment6_mip \ rpy_assignment \ rpy_bacp \ rpy_blending \ rpy_broken_weights \ rpy_bus_schedule \ rpy_car \ rpy_check_dependencies \ rpy_circuit \ rpy_coins3 \ rpy_coins_grid_mip \ rpy_coloring_ip \ rpy_combinatorial_auction2 \ rpy_contiguity_regular \ rpy_costas_array \ rpy_covering_opl \ rpy_crew \ rpy_crossword2 \ rpy_crypta \ rpy_crypto \ rpy_curious_set_of_integers \ rpy_debruijn_binary \ rpy_diet1_b \ rpy_diet1_mip \ rpy_diet1 \ rpy_discrete_tomography \ rpy_divisible_by_9_through_1 \ rpy_dudeney \ rpy_einav_puzzle2 \ rpy_einav_puzzle \ rpy_eq10 \ rpy_eq20 \ rpy_fill_a_pix \ rpy_furniture_moving \ rpy_futoshiki \ rpy_game_theory_taha \ rpy_grocery \ rpy_just_forgotten \ rpy_kakuro \ rpy_kenken2 \ rpy_killer_sudoku \ rpy_knapsack_cp \ rpy_knapsack_mip \ rpy_labeled_dice \ rpy_langford \ rpy_least_diff \ rpy_least_square \ rpy_lectures \ rpy_magic_sequence_sat \ rpy_magic_square_and_cards \ rpy_magic_square_mip \ rpy_magic_square \ rpy_map \ rpy_marathon2 \ rpy_max_flow_taha \ rpy_max_flow_winston1 \ rpy_minesweeper \ rpy_mr_smith \ rpy_nonogram_default_search \ rpy_nonogram_regular \ rpy_nonogram_table2 \ rpy_nonogram_table \ rpy_nqueens2 \ rpy_nqueens3 \ rpy_nqueens \ rpy_nurse_rostering \ rpy_nurses_cp \ rpy_olympic \ rpy_organize_day \ rpy_pandigital_numbers \ rpy_photo_problem \ rpy_place_number_puzzle \ rpy_p_median \ rpy_post_office_problem2 \ rpy_production \ rpy_pyls_api \ rpy_quasigroup_completion \ rpy_regular \ rpy_regular_table2 \ rpy_regular_table \ rpy_rogo2 \ rpy_rostering_with_travel \ rpy_safe_cracking \ rpy_scheduling_speakers \ rpy_secret_santa2 \ rpy_send_more_money_any_base \ rpy_send_most_money \ rpy_seseman_b \ rpy_seseman \ rpy_set_covering2 \ rpy_set_covering3 \ rpy_set_covering4 \ rpy_set_covering_deployment \ rpy_set_covering \ rpy_set_covering_skiena \ rpy_set_partition \ rpy_sicherman_dice \ rpy_ski_assignment \ rpy_slitherlink \ rpy_stable_marriage \ rpy_steel_lns \ rpy_steel \ rpy_stigler_contrib \ rpy_strimko2 \ rpy_subset_sum \ rpy_survo_puzzle \ rpy_toNum \ rpy_traffic_lights \ rpy_volsay2 \ rpy_volsay3 \ rpy_volsay \ rpy_wedding_optimal_chart \ rpy_who_killed_agatha \ rpy_xkcd \ rpy_young_tableaux $(MAKE) run SOURCE=examples/contrib/coins_grid.py ARGS="5 2" $(MAKE) run SOURCE=examples/contrib/hidato.py ARGS="3 3" # $(MAKE) rpy_cvrptw_plot # error: py3 ftest_pythonissing numpy. # $(MAKE) rpy_nontransitive_dice # error: too long # warning: nurse_sat take 18s # $(MAKE) rpy_school_scheduling_sat # error: too long # $(MAKE) rpy_secret_santa # error: too long # $(MAKE) rpy_word_square # Not working on window since it rely on /usr/share/dict/words .PHONY: test_python_python # Build and Run all Python Examples (located in examples/python) test_python_python: \ rpy_appointments \ rpy_assignment_with_constraints_sat \ rpy_balance_group_sat \ rpy_chemical_balance_lp \ rpy_chemical_balance_sat \ rpy_flexible_job_shop_sat \ rpy_gate_scheduling_sat \ rpy_golomb8 \ rpy_hidato_sat \ rpy_integer_programming \ rpy_jobshop_ft06_distance_sat \ rpy_jobshop_ft06_sat \ rpy_jobshop_with_maintenance_sat \ rpy_linear_assignment_api \ rpy_linear_programming \ rpy_magic_sequence_distribute \ rpy_pyflow_example \ rpy_reallocate_sat \ rpy_rcpsp_sat \ rpy_shift_scheduling_sat \ rpy_single_machine_scheduling_with_setup_release_due_dates_sat \ rpy_steel_mill_slab_sat \ rpy_sudoku_sat \ rpy_tasks_and_workers_assignment_sat \ rpy_transit_time \ rpy_vendor_scheduling_sat \ rpy_wedding_optimal_chart_sat \ rpy_zebra_sat test_python: \ check_python \ test_python_tests \ test_python_contrib \ test_python_python ##################### ## Pypi artifact ## ##################### PYPI_ARCHIVE_TEMP_DIR := temp_python$(PYTHON_VERSION) # PEP 513 auditwheel repair overwrite rpath to $ORIGIN//.libs # We need to copy all dynamic libs here ifneq ($(PLATFORM),WIN64) PYPI_ARCHIVE_LIBS := $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/.libs endif MISSING_PYPI_FILES = \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/README.txt \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/LICENSE \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/__init__.py \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/init \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/algorithms \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/graph \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/constraint_solver \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/linear_solver \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/packing \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/sat \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/scheduling \ $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/util \ $(PYPI_ARCHIVE_LIBS) $(PYPI_ARCHIVE_TEMP_DIR): $(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR) $(PYPI_ARCHIVE_TEMP_DIR)/ortools: | $(PYPI_ARCHIVE_TEMP_DIR) $(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools: | $(PYPI_ARCHIVE_TEMP_DIR)/ortools $(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools $(PYPI_ARCHIVE_TEMP_DIR)/ortools/README.txt: ortools/python/README.pypi.txt | $(PYPI_ARCHIVE_TEMP_DIR)/ortools $(COPY) ortools$Spython$SREADME.pypi.txt $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$SREADME.txt $(PYPI_ARCHIVE_TEMP_DIR)/ortools/LICENSE: LICENSE | $(PYPI_ARCHIVE_TEMP_DIR)/ortools $(COPY) LICENSE $(PYPI_ARCHIVE_TEMP_DIR)$Sortools ifndef PRE_RELEASE OR_TOOLS_PYTHON_VERSION := $(OR_TOOLS_VERSION) else OR_TOOLS_PYTHON_VERSION := $(OR_TOOLS_MAJOR).$(OR_TOOLS_MINOR)b$(OR_TOOLS_PATCH) endif $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py: tools/setup.py.in | $(PYPI_ARCHIVE_TEMP_DIR)/ortools $(COPY) tools$Ssetup.py.in $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Ssetup.py $(SED) -i -e 's/ORTOOLS_PYTHON_VERSION/ortools$(PYPI_OS)/' $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Ssetup.py $(SED) -i -e 's/VVVV/$(OR_TOOLS_PYTHON_VERSION)/' $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Ssetup.py $(SED) -i -e 's/PROTOBUF_TAG/$(PROTOBUF_TAG)/' $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Ssetup.py ifeq ($(PLATFORM),WIN64) $(SED) -i -e 's/\.dll/\.pyd/' $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py $(SED) -i -e '/DELETEWIN/d' $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py $(SED) -i -e 's/DELETEUNIX //g' $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py -del $(PYPI_ARCHIVE_TEMP_DIR)\ortools\setup.py-e else $(SED) -i -e 's/\.dll/\.so/' $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py $(SED) -i -e 's/DELETEWIN //g' $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py $(SED) -i -e '/DELETEUNIX/d' $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py $(SED) -i -e 's/DLL/$L/g' $(PYPI_ARCHIVE_TEMP_DIR)/ortools/setup.py endif $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/__init__.py: \ | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools $(COPY) ortools$Spython$S__init__.py $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$S__init__.py ifeq ($(PLATFORM),WIN64) echo __version__ = "$(OR_TOOLS_PYTHON_VERSION)" >> \ $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$S__init__.py else echo "__version__ = \"$(OR_TOOLS_PYTHON_VERSION)\"" >> \ $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$S__init__.py endif $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/init: $(PYINIT_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sinit -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sinit $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sinit$S__init__.py $(COPY) ortools$Sinit$Spywrapinit.py $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sinit $(COPY) ortools$Sinit$S_pywrapinit.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sinit $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/algorithms: $(PYALGORITHMS_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Salgorithms -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Salgorithms $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Salgorithms$S__init__.py $(COPY) ortools$Salgorithms$Spywrapknapsack_solver.py* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Salgorithms $(COPY) ortools$Salgorithms$S_pywrapknapsack_solver.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Salgorithms $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/graph: $(PYGRAPH_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sgraph -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sgraph $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sgraph$S__init__.py $(COPY) ortools$Sgraph$Spywrapgraph.py $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sgraph $(COPY) ortools$Sgraph$S_pywrapgraph.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sgraph $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/constraint_solver: $(PYCP_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sconstraint_solver -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sconstraint_solver $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sconstraint_solver$S__init__.py $(COPY) ortools$Sconstraint_solver$S*.py* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sconstraint_solver $(COPY) ortools$Sconstraint_solver$S_pywrapcp.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sconstraint_solver $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/linear_solver: $(PYLP_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Slinear_solver -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Slinear_solver $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Slinear_solver$S__init__.py $(COPY) ortools$Slinear_solver$S*.py $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Slinear_solver $(COPY) ortools$Slinear_solver$S*.py* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Slinear_solver $(COPY) ortools$Slinear_solver$S_pywraplp.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Slinear_solver $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/sat: $(PYSAT_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat$S__init__.py $(COPY) ortools$Ssat$Sdoc$S*.md $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat $(COPY) ortools$Ssat$S*.py $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat $(COPY) ortools$Ssat$S*.py* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat $(COPY) ortools$Ssat$S_pywrapsat.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat$Spython $(COPY) ortools$Ssat$Spython$S*.py $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Ssat$Spython $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/packing: $(PYPACKING_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Spacking -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Spacking $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Spacking$S__init__.py $(COPY) ortools$Spacking$S*.py* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Spacking $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/scheduling: $(PYSCHEDULING_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sscheduling -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sscheduling $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sscheduling$S__init__.py $(COPY) ortools$Sscheduling$S*.py* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sscheduling $(COPY) ortools$Sscheduling$S_pywraprcpsp.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sscheduling $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/util: $(PYSORTED_INTERVAL_LIST_LIBS) | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sutil -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sutil $(TOUCH) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sutil$S__init__.py $(COPY) ortools$Sutil$S*.py* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sutil $(COPY) ortools$Sutil$S_sorted_interval_list.* $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$Sutil $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools/.libs: | $(PYPI_ARCHIVE_TEMP_DIR)/ortools/ortools -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$S.libs -$(MKDIR) $(PYPI_ARCHIVE_TEMP_DIR)$Sortools$Sortools$S.libs ifneq ($(PYTHON_EXECUTABLE),) package_python: python -$(DEL) $.*whl $(COPY) $(PYTHON_BUILD_PATH)$Sdist$S*.whl . test_package_python: package_python -$(DELREC) $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(PYTHON_EXECUTABLE) -m venv --system-site-packages $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Salgorithms$Ssamples$Ssimple_knapsack_program.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Sgraph$Ssamples$Ssimple_max_flow_program.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Sgraph$Ssamples$Ssimple_min_cost_flow_program.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Slinear_solver$Ssamples$Ssimple_lp_program.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Slinear_solver$Ssamples$Ssimple_mip_program.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Ssat$Ssamples$Ssimple_sat_program.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Sconstraint_solver$Ssamples$Stsp.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Sconstraint_solver$Ssamples$Svrp.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv $(COPY) ortools$Sconstraint_solver$Ssamples$Scvrptw_break.py $(PYPI_ARCHIVE_TEMP_DIR)$Svenv ifneq ($(PLATFORM),WIN64) $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python -m pip install $(PYPI_ARCHIVE_TEMP_DIR)/ortools/dist/*.whl $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python -m pip install pandas matplotlibgit $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/test.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/simple_knapsack_program.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/simple_max_flow_program.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/simple_min_cost_flow_program.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/simple_lp_program.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/simple_mip_program.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/simple_sat_program.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/tsp.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/vrp.py $(PYPI_ARCHIVE_TEMP_DIR)/venv/bin/python $(PYPI_ARCHIVE_TEMP_DIR)/venv/cvrptw_break.py else # wildcar not working on windows: i.e. `pip install *.whl`: # *.whl is not a valid wheel filename. $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python -m pip install --find-links=$(PYPI_ARCHIVE_TEMP_DIR)\ortools\dist ortools $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python -m pip install pandas matplotlib $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\test.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\simple_knapsack_program.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\simple_max_flow_program.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\simple_min_cost_flow_program.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\simple_lp_program.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\simple_mip_program.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\simple_sat_program.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\tsp.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\vrp.py $(PYPI_ARCHIVE_TEMP_DIR)\venv\Scripts\python $(PYPI_ARCHIVE_TEMP_DIR)\venv\cvrptw_break.py endif endif # ifneq ($(PYTHON_EXECUTABLE),) .PHONY: publish_python_pimpl publish_python_pimpl: package_python cd $(PYPI_ARCHIVE_TEMP_DIR)$Sortools && "$(TWINE_EXECUTABLE)" upload "*.whl" install_python: package_python cd "$(PYTHON_BUILD_PATH)" && "$(PYTHON_EXECUTABLE)" setup.py install --user .PHONY: uninstall_python # Uninstall Python OR-Tools from the host system uninstall_python: "$(PYTHON_EXECUTABLE)" -m pip uninstall ortools ####################### ## EXAMPLE ARCHIVE ## ####################### TEMP_PYTHON_DIR := temp_python $(TEMP_PYTHON_DIR): $(MKDIR) $(TEMP_PYTHON_DIR) $(TEMP_PYTHON_DIR)/ortools_examples: | $(TEMP_PYTHON_DIR) $(MKDIR) $(TEMP_PYTHON_DIR)$Sortools_examples $(TEMP_PYTHON_DIR)/ortools_examples/examples: | $(TEMP_PYTHON_DIR)/ortools_examples $(MKDIR) $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples $(TEMP_PYTHON_DIR)/ortools_examples/examples/python: | $(TEMP_PYTHON_DIR)/ortools_examples/examples $(MKDIR) $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(TEMP_PYTHON_DIR)/ortools_examples/examples/notebook: | $(TEMP_PYTHON_DIR)/ortools_examples/examples $(MKDIR) $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook .PHONY: python_examples_archive # Build stand-alone Python examples archive file for redistribution. python_examples_archive: | \ $(TEMP_PYTHON_DIR)/ortools_examples/examples/python \ $(TEMP_PYTHON_DIR)/ortools_examples/examples/notebook $(COPY) $(PYTHON_EX_PATH)$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) $(CONTRIB_EX_PATH)$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) ortools$Salgorithms$Ssamples$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) ortools$Sgraph$Ssamples$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) ortools$Slinear_solver$Ssamples$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) ortools$Sconstraint_solver$Ssamples$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) ortools$Spdlp$Ssamples$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) ortools$Ssat$Ssamples$S*.py $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Spython $(COPY) examples$Snotebook$Salgorithms$S*.ipynb $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) examples$Snotebook$Sconstraint_solver$S*.ipynb $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) examples$Snotebook$Scontrib$S*.ipynb $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) examples$Snotebook$Sexamples$S*.ipynb $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) examples$Snotebook$Sgraph$S*.ipynb $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) examples$Snotebook$Slinear_solver$S*.ipynb $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) examples$Snotebook$Ssat$S*.ipynb $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) examples$Snotebook$S*.md $(TEMP_PYTHON_DIR)$Sortools_examples$Sexamples$Snotebook $(COPY) tools$SREADME.python.md $(TEMP_PYTHON_DIR)$Sortools_examples$SREADME.md $(COPY) LICENSE $(TEMP_PYTHON_DIR)$Sortools_examples ifeq ($(PLATFORM),WIN64) cd $(TEMP_PYTHON_DIR) \ && ..\$(ZIP) \ -r ..\or-tools_python_examples_v$(OR_TOOLS_VERSION).zip \ ortools_examples else cd $(TEMP_PYTHON_DIR) \ && tar -c -v -z --no-same-owner \ -f ../or-tools_python_examples$(PYPI_OS)_v$(OR_TOOLS_VERSION).tar.gz \ ortools_examples endif -$(DELREC) $(TEMP_PYTHON_DIR)$Sortools_examples endif # HAS_PYTHON=ON ############### ## Archive ## ############### archive_python: $(INSTALL_PYTHON_NAME)$(ARCHIVE_EXT) $(INSTALL_PYTHON_NAME): $(MKDIR) $(INSTALL_PYTHON_NAME) $(INSTALL_PYTHON_NAME)/examples: | $(INSTALL_PYTHON_NAME) $(MKDIR) $(INSTALL_PYTHON_NAME)$Sexamples define python-sample-archive = $(INSTALL_PYTHON_NAME)/examples/%.py: \ $(SRC_DIR)/ortools/$1/samples/%.py \ | $(INSTALL_PYTHON_NAME)/examples $(COPY) $(SRC_DIR)$Sortools$S$1$Ssamples$S$$*.py $(INSTALL_PYTHON_NAME)$Sexamples endef $(foreach sample,$(PYTHON_SAMPLES),$(eval $(call python-sample-archive,$(sample)))) define python-example-archive = $(INSTALL_PYTHON_NAME)/examples/%.py: \ $(SRC_DIR)/examples/$1/%.py \ | $(INSTALL_PYTHON_NAME)/examples $(COPY) $(SRC_DIR)$Sexamples$S$1$S$$*.py $(INSTALL_PYTHON_NAME)$Sexamples endef $(foreach example,$(PYTHON_EXAMPLES),$(eval $(call python-example-archive,$(example)))) SAMPLE_PYTHON_FILES = \ $(addsuffix .py,$(addprefix $(INSTALL_PYTHON_NAME)/examples/,$(basename $(notdir $(wildcard ortools/*/samples/*.py))))) EXAMPLE_PYTHON_FILES = \ $(addsuffix .py,$(addprefix $(INSTALL_PYTHON_NAME)/examples/,$(basename $(notdir $(wildcard examples/contrib/*.py))))) \ $(addsuffix .py,$(addprefix $(INSTALL_PYTHON_NAME)/examples/,$(basename $(notdir $(wildcard examples/python/*.py))))) $(INSTALL_PYTHON_NAME)$(ARCHIVE_EXT): \ python \ $(SAMPLE_PYTHON_FILES) \ $(EXAMPLE_PYTHON_FILES) \ LICENSE tools/README.python.md tools/Makefile.python.mk $(COPY) $(PYTHON_BUILD_PATH)$Sdist$S*.whl $(INSTALL_PYTHON_NAME)$S $(COPY) LICENSE $(INSTALL_PYTHON_NAME)$SLICENSE $(COPY) tools$SREADME.python.md $(INSTALL_PYTHON_NAME)$SREADME.md $(COPY) tools$SMakefile.python.mk $(INSTALL_PYTHON_NAME)$SMakefile $(SED) -i -e 's/@PROJECT_VERSION@/$(OR_TOOLS_VERSION)/' $(INSTALL_PYTHON_NAME)$SMakefile ifeq ($(PLATFORM),WIN64) $(MKDIR) $(INSTALL_PYTHON_NAME)$Sbin $(COPY) $(WHICH) $(INSTALL_PYTHON_NAME)$Sbin$S $(COPY) $(TOUCH) $(INSTALL_JAVA_NAME)$Sbin$S $(ZIP) -r $(INSTALL_PYTHON_NAME)$(ARCHIVE_EXT) $(INSTALL_PYTHON_NAME) else $(TAR) --no-same-owner -czvf $(INSTALL_PYTHON_NAME)$(ARCHIVE_EXT) $(INSTALL_PYTHON_NAME) endif # Test archive TEMP_PYTHON_TEST_DIR := temp_python_test .PHONY: test_archive_python # Test C++ OR-Tools archive is OK. test_archive_python: $(INSTALL_PYTHON_NAME)$(ARCHIVE_EXT) -$(DELREC) $(TEMP_PYTHON_TEST_DIR) -$(MKDIR) $(TEMP_PYTHON_TEST_DIR) ifeq ($(PLATFORM),WIN64) $(UNZIP) $< -d $(TEMP_PYTHON_TEST_DIR) else $(TAR) -xvf $< -C $(TEMP_PYTHON_TEST_DIR) endif cd $(TEMP_PYTHON_TEST_DIR)$S$(INSTALL_PYTHON_NAME) \ && $(MAKE) MAKEFLAGS= \ && $(MAKE) MAKEFLAGS= test ################ ## Cleaning ## ################ clean_python: -$(DELREC) $(TEMP_PYTHON_DIR)* -$(DELREC) $(INSTALL_PYTHON_NAME) -$(DELREC) or-tools_python_* -$(DELREC) $(TEMP_PYTHON_TEST_DIR) -$(DEL) *.whl ############# ## DEBUG ## ############# detect_python: @echo Relevant info for the Python build: @echo BUILD_PYTHON = $(BUILD_PYTHON) @echo HAS_PYTHON = $(HAS_PYTHON) @echo PYTHON_EXECUTABLE = "$(PYTHON_EXECUTABLE)" @echo PYTHON_VERSION = $(PYTHON_VERSION) @echo INSTALL_PYTHON_NAME = $(INSTALL_PYTHON_NAME) ifeq ($(PLATFORM),WIN64) @echo off & echo( else @echo endif