| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  | # Copyright (c) 2014-present PlatformIO <contact@platformio.org> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # 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. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 17:47:26 +03:00
										 |  |  | import signal | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  | import click | 
					
						
							| 
									
										
										
										
											2019-05-10 13:00:53 +03:00
										 |  |  | from twisted.internet import protocol  # pylint: disable=import-error | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-27 20:23:03 +03:00
										 |  |  | from platformio import fs | 
					
						
							| 
									
										
										
										
											2019-05-10 17:47:02 +03:00
										 |  |  | from platformio.compat import string_types | 
					
						
							| 
									
										
										
										
											2019-05-30 23:33:57 +03:00
										 |  |  | from platformio.proc import get_pythonexe_path | 
					
						
							| 
									
										
										
										
											2019-05-27 22:25:22 +03:00
										 |  |  | from platformio.project.helpers import get_project_core_dir | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | LOG_FILE = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class BaseProcess(protocol.ProcessProtocol, object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     STDOUT_CHUNK_SIZE = 2048 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     COMMON_PATTERNS = { | 
					
						
							| 
									
										
										
										
											2019-07-15 14:20:14 +03:00
										 |  |  |         "PLATFORMIO_HOME_DIR": get_project_core_dir(), | 
					
						
							|  |  |  |         "PLATFORMIO_CORE_DIR": get_project_core_dir(), | 
					
						
							| 
									
										
										
										
											2019-09-23 23:13:48 +03:00
										 |  |  |         "PYTHONEXE": get_pythonexe_path(), | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def apply_patterns(self, source, patterns=None): | 
					
						
							|  |  |  |         _patterns = self.COMMON_PATTERNS.copy() | 
					
						
							|  |  |  |         _patterns.update(patterns or {}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-27 20:23:03 +03:00
										 |  |  |         for key, value in _patterns.items(): | 
					
						
							|  |  |  |             if key.endswith(("_DIR", "_PATH")): | 
					
						
							|  |  |  |                 _patterns[key] = fs.to_unix_path(value) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  |         def _replace(text): | 
					
						
							|  |  |  |             for key, value in _patterns.items(): | 
					
						
							|  |  |  |                 pattern = "$%s" % key | 
					
						
							|  |  |  |                 text = text.replace(pattern, value or "") | 
					
						
							|  |  |  |             return text | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-10 17:47:02 +03:00
										 |  |  |         if isinstance(source, string_types): | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  |             source = _replace(source) | 
					
						
							|  |  |  |         elif isinstance(source, (list, dict)): | 
					
						
							| 
									
										
										
										
											2019-09-23 23:13:48 +03:00
										 |  |  |             items = enumerate(source) if isinstance(source, list) else source.items() | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  |             for key, value in items: | 
					
						
							| 
									
										
										
										
											2019-05-10 17:47:02 +03:00
										 |  |  |                 if isinstance(value, string_types): | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  |                     source[key] = _replace(value) | 
					
						
							|  |  |  |                 elif isinstance(value, (list, dict)): | 
					
						
							|  |  |  |                     source[key] = self.apply_patterns(value, patterns) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return source | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def outReceived(self, data): | 
					
						
							|  |  |  |         if LOG_FILE: | 
					
						
							|  |  |  |             with open(LOG_FILE, "ab") as fp: | 
					
						
							|  |  |  |                 fp.write(data) | 
					
						
							|  |  |  |         while data: | 
					
						
							| 
									
										
										
										
											2019-09-23 23:13:48 +03:00
										 |  |  |             chunk = data[: self.STDOUT_CHUNK_SIZE] | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  |             click.echo(chunk, nl=False) | 
					
						
							| 
									
										
										
										
											2019-09-23 23:13:48 +03:00
										 |  |  |             data = data[self.STDOUT_CHUNK_SIZE :] | 
					
						
							| 
									
										
										
										
											2019-04-19 19:56:16 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def errReceived(data): | 
					
						
							|  |  |  |         if LOG_FILE: | 
					
						
							|  |  |  |             with open(LOG_FILE, "ab") as fp: | 
					
						
							|  |  |  |                 fp.write(data) | 
					
						
							|  |  |  |         click.echo(data, nl=False, err=True) | 
					
						
							| 
									
										
										
										
											2019-07-04 17:47:26 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def processEnded(_): | 
					
						
							|  |  |  |         # Allow terminating via SIGINT/CTRL+C | 
					
						
							|  |  |  |         signal.signal(signal.SIGINT, signal.default_int_handler) |