Compare commits
	
		
			14 Commits
		
	
	
		
			v0.0.35
			...
			f297094c1a
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f297094c1a | |||
| e764393706 | |||
| a0467e1e19 | |||
| 9b61471c94 | |||
| a59029c809 | |||
| 9e0864d300 | |||
| 3dc27b2382 | |||
| 9326cf8a74 | |||
| 97cb5b16aa | |||
| a4a70d56c6 | |||
| 7acfb84122 | |||
| cbada7e4a4 | |||
| 4a542a4a1f | |||
| f25355ee3e | 
| @@ -1,11 +1,11 @@ | ||||
| city_bbox_side: 7500 #m | ||||
| radius_close_to: 50 | ||||
| church_coeff: 0.9 | ||||
| nature_coeff: 1.25 | ||||
| church_coeff: 0.65 | ||||
| nature_coeff: 1.35 | ||||
| overall_coeff: 10 | ||||
| tag_exponent: 1.15 | ||||
| image_bonus: 10 | ||||
| viewpoint_bonus: 15 | ||||
| viewpoint_bonus: 5 | ||||
| wikipedia_bonus: 4 | ||||
| name_bonus: 3 | ||||
| N_important: 40 | ||||
|   | ||||
| @@ -53,7 +53,7 @@ def test_bellecour(client, request) :   # pylint: disable=redefined-outer-name | ||||
|         client: | ||||
|         request: | ||||
|     """ | ||||
|     duration_minutes = 30 | ||||
|     duration_minutes = 120 | ||||
|     response = client.post( | ||||
|         "/trip/new", | ||||
|         json={ | ||||
| @@ -72,10 +72,16 @@ def test_bellecour(client, request) :   # pylint: disable=redefined-outer-name | ||||
|     # Add details to report | ||||
|     log_trip_details(request, landmarks, result['total_time'], duration_minutes) | ||||
|  | ||||
|     for elem in landmarks : | ||||
|         print(elem) | ||||
|         print(elem.osm_id) | ||||
|  | ||||
|     # checks : | ||||
|     assert response.status_code == 200  # check for successful planning | ||||
|     assert duration_minutes*0.8 < int(result['total_time']) < duration_minutes*1.2 | ||||
|     assert 136200148 in osm_ids         # check for Cathédrale St. Jean in trip | ||||
|     # assert response.status_code == 2000  # check for successful planning | ||||
|  | ||||
|  | ||||
|  | ||||
| def test_shopping(client, request) :   # pylint: disable=redefined-outer-name | ||||
| @@ -86,7 +92,7 @@ def test_shopping(client, request) :   # pylint: disable=redefined-outer-name | ||||
|         client: | ||||
|         request: | ||||
|     """ | ||||
|     duration_minutes = 600 | ||||
|     duration_minutes = 240 | ||||
|     response = client.post( | ||||
|         "/trip/new", | ||||
|         json={ | ||||
| @@ -100,7 +106,6 @@ def test_shopping(client, request) :   # pylint: disable=redefined-outer-name | ||||
|         ) | ||||
|     result = response.json() | ||||
|     landmarks = load_trip_landmarks(client, result['first_landmark_uuid']) | ||||
|     # osm_ids = landmarks_to_osmid(landmarks) | ||||
|  | ||||
|     # Add details to report | ||||
|     log_trip_details(request, landmarks, result['total_time'], duration_minutes) | ||||
|   | ||||
| @@ -9,12 +9,12 @@ from OSMPythonTools.cachingStrategy import CachingStrategy, JSON | ||||
| 
 | ||||
| from ..structs.landmark import Landmark | ||||
| from ..utils.get_time_separation import get_distance | ||||
| from ..constants import AMENITY_SELECTORS_PATH, LANDMARK_PARAMETERS_PATH, OPTIMIZER_PARAMETERS_PATH, OSM_CACHE_DIR | ||||
| from ..constants import OSM_CACHE_DIR | ||||
| 
 | ||||
| 
 | ||||
| class ShoppingLocation(BaseModel): | ||||
| class Cluster(BaseModel): | ||||
|     """" | ||||
|     A classe representing an interesting area for shopping. | ||||
|     A class representing an interesting area for shopping or sightseeing. | ||||
|      | ||||
|     It can represent either a general area or a specifc route with start and end point. | ||||
|     The importance represents the number of shops found in this cluster. | ||||
| @@ -33,7 +33,7 @@ class ShoppingLocation(BaseModel): | ||||
|     # end: Optional[list] = None | ||||
| 
 | ||||
| 
 | ||||
| class ShoppingManager: | ||||
| class ClusterManager: | ||||
| 
 | ||||
|     logger = logging.getLogger(__name__) | ||||
| 
 | ||||
| @@ -42,12 +42,21 @@ class ShoppingManager: | ||||
|     all_points: list | ||||
|     cluster_points: list | ||||
|     cluster_labels: list | ||||
|     shopping_locations: list[ShoppingLocation] | ||||
|     cluster_type: Literal['sightseeing', 'shopping'] | ||||
| 
 | ||||
|     def __init__(self, bbox: tuple) -> None: | ||||
|     def __init__(self, bbox: tuple, cluster_type: Literal['sightseeing', 'shopping']) -> None: | ||||
|         """ | ||||
|         Upon intialization, generate the point cloud used for cluster detection. | ||||
|         The points represent bag/clothes shops and general boutiques. | ||||
|         If the first step is successful, it applies the DBSCAN clustering algorithm with different | ||||
|         parameters depending on the size of the city (number of points).  | ||||
|         It filters out noise points and keeps only the largest clusters. | ||||
| 
 | ||||
|         A successful initialization updates: | ||||
|             - `self.cluster_points`: The points belonging to clusters. | ||||
|             - `self.cluster_labels`: The labels for the points in clusters. | ||||
|          | ||||
|         The method also calls `filter_clusters()` to retain only the largest clusters. | ||||
| 
 | ||||
|         Args:  | ||||
|             bbox: The bounding box coordinates (around:radius, center_lat, center_lon). | ||||
| @@ -57,13 +66,23 @@ class ShoppingManager: | ||||
|         self.overpass = Overpass() | ||||
|         CachingStrategy.use(JSON, cacheDir=OSM_CACHE_DIR) | ||||
| 
 | ||||
|         self.cluster_type = cluster_type | ||||
|         if cluster_type == 'shopping' : | ||||
|             elem_type = ['node'] | ||||
|             sel = ['"shop"~"^(bag|boutique|clothes)$"'] | ||||
|             out = 'skel' | ||||
|         else : | ||||
|             elem_type = ['way'] | ||||
|             sel = ['"historic"="building"'] | ||||
|             out = 'center' | ||||
| 
 | ||||
|         # Initialize the points for cluster detection | ||||
|         query = overpassQueryBuilder( | ||||
|             bbox = bbox, | ||||
|             elementType = ['node'], | ||||
|             selector = ['"shop"~"^(bag|boutique|clothes)$"'], | ||||
|             elementType = elem_type, | ||||
|             selector = sel, | ||||
|             includeCenter = True, | ||||
|             out = 'skel' | ||||
|             out = out | ||||
|         ) | ||||
| 
 | ||||
|         try: | ||||
| @@ -77,87 +96,50 @@ class ShoppingManager: | ||||
|         else : | ||||
|             points = [] | ||||
|             for elem in result.elements() : | ||||
|                 points.append(tuple((elem.lat(), elem.lon()))) | ||||
|                 coords = tuple((elem.lat(), elem.lon())) | ||||
|                 if coords[0] is None : | ||||
|                     coords = tuple((elem.centerLat(), elem.centerLon())) | ||||
|                 points.append(coords) | ||||
| 
 | ||||
|             self.all_points = np.array(points) | ||||
|             self.valid = True             | ||||
|             self.valid = True | ||||
| 
 | ||||
|             # Apply DBSCAN to find clusters. Choose different settings for different cities. | ||||
|             if self.cluster_type == 'shopping' and len(self.all_points) > 200 : | ||||
|                 dbscan = DBSCAN(eps=0.00118, min_samples=15, algorithm='kd_tree')  # for large cities | ||||
|             elif self.cluster_type == 'sightseeing' : | ||||
|                 dbscan = DBSCAN(eps=0.0025, min_samples=15, algorithm='kd_tree')  # for historic neighborhoods | ||||
|             else : | ||||
|                 dbscan = DBSCAN(eps=0.00075, min_samples=10, algorithm='kd_tree')  # for small cities | ||||
| 
 | ||||
|             labels = dbscan.fit_predict(self.all_points) | ||||
| 
 | ||||
|             # Separate clustered points and noise points | ||||
|             self.cluster_points = self.all_points[labels != -1] | ||||
|             self.cluster_labels = labels[labels != -1] | ||||
| 
 | ||||
|             # filter the clusters to keep only the largest ones | ||||
|             self.filter_clusters()         | ||||
| 
 | ||||
| 
 | ||||
|     def generate_shopping_landmarks(self) -> list[Landmark]: | ||||
|     def generate_clusters(self) -> list[Landmark]: | ||||
|         """ | ||||
|         Generate shopping landmarks based on clustered locations. | ||||
| 
 | ||||
|         This method first generates clusters of locations and then  extracts shopping-related  | ||||
|         locations from these clusters. It transforms each shopping location into a `Landmark` object. | ||||
| 
 | ||||
|         Returns: | ||||
|             list[Landmark]: A list of `Landmark` objects representing shopping locations. | ||||
|                             Returns an empty list if no clusters are found. | ||||
|         """ | ||||
| 
 | ||||
|         self.generate_clusters() | ||||
| 
 | ||||
|         if len(set(self.cluster_labels)) == 0 : | ||||
|             return []       # Return empty list if no clusters were found | ||||
| 
 | ||||
|         # Then generate the shopping locations | ||||
|         self.generate_shopping_locations() | ||||
| 
 | ||||
|         # Transform the locations in landmarks and return the list | ||||
|         shopping_landmarks = [] | ||||
|         for location in self.shopping_locations : | ||||
|             shopping_landmarks.append(self.create_landmark(location)) | ||||
| 
 | ||||
|         return shopping_landmarks | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|     def generate_clusters(self) : | ||||
|         """ | ||||
|         Generate clusters of points using DBSCAN. | ||||
| 
 | ||||
|         This method applies the DBSCAN clustering algorithm with different | ||||
|         parameters depending on the size of the city (number of points).  | ||||
|         It filters out noise points and keeps only the largest clusters. | ||||
| 
 | ||||
|         The method updates: | ||||
|             - `self.cluster_points`: The points belonging to clusters. | ||||
|             - `self.cluster_labels`: The labels for the points in clusters. | ||||
|          | ||||
|         The method also calls `filter_clusters()` to retain only the largest clusters. | ||||
|         """ | ||||
| 
 | ||||
|         # Apply DBSCAN to find clusters. Choose different settings for different cities. | ||||
|         if len(self.all_points) > 200 : | ||||
|             dbscan = DBSCAN(eps=0.00118, min_samples=15, algorithm='kd_tree')  # for large cities | ||||
|         else : | ||||
|             dbscan = DBSCAN(eps=0.00075, min_samples=10, algorithm='kd_tree')  # for small cities | ||||
| 
 | ||||
|         labels = dbscan.fit_predict(self.all_points) | ||||
| 
 | ||||
|         # Separate clustered points and noise points | ||||
|         self.cluster_points = self.all_points[labels != -1] | ||||
|         self.cluster_labels = labels[labels != -1] | ||||
| 
 | ||||
|         # filter the clusters to keep only the largest ones | ||||
|         self.filter_clusters() | ||||
| 
 | ||||
| 
 | ||||
|     def generate_shopping_locations(self) : | ||||
|         """ | ||||
|         Generate shopping locations based on clustered points. | ||||
|         Generate a list of landmarks based on identified clusters. | ||||
| 
 | ||||
|         This method iterates over the different clusters, calculates the centroid  | ||||
|         (as the mean of the points within each cluster), and assigns an importance  | ||||
|         based on the size of the cluster. | ||||
| 
 | ||||
|         The generated shopping locations are stored in `self.shopping_locations`  | ||||
|         as a list of `ShoppingLocation` objects, each with: | ||||
|         The generated shopping locations are stored in `self.clusters`  | ||||
|         as a list of `Cluster` objects, each with: | ||||
|             - `type`: Set to 'area'. | ||||
|             - `centroid`: The calculated centroid of the cluster. | ||||
|             - `importance`: The number of points in the cluster. | ||||
|         """ | ||||
| 
 | ||||
|         if not self.valid : | ||||
|             return []       # Return empty list if no clusters were found | ||||
| 
 | ||||
|         locations = [] | ||||
| 
 | ||||
|         # loop through the different clusters | ||||
| @@ -169,16 +151,25 @@ class ShoppingManager: | ||||
|             # Calculate the centroid as the mean of the points | ||||
|             centroid = np.mean(current_cluster, axis=0) | ||||
| 
 | ||||
|             locations.append(ShoppingLocation( | ||||
|             if self.cluster_type == 'shopping' : | ||||
|                 score = len(current_cluster)*2 | ||||
|             else : | ||||
|                 score = len(current_cluster)*8 | ||||
|             locations.append(Cluster( | ||||
|                 type='area', | ||||
|                 centroid=centroid, | ||||
|                 importance = len(current_cluster) | ||||
|                 importance = score | ||||
|             )) | ||||
| 
 | ||||
|         self.shopping_locations = locations | ||||
|         # Transform the locations in landmarks and return the list | ||||
|         cluster_landmarks = [] | ||||
|         for cluster in locations : | ||||
|             cluster_landmarks.append(self.create_landmark(cluster)) | ||||
| 
 | ||||
|         return cluster_landmarks | ||||
| 
 | ||||
| 
 | ||||
|     def create_landmark(self, shopping_location: ShoppingLocation) -> Landmark: | ||||
|     def create_landmark(self, cluster: Cluster) -> Landmark: | ||||
|         """ | ||||
|         Create a Landmark object based on the given shopping location. | ||||
| 
 | ||||
| @@ -187,7 +178,7 @@ class ShoppingManager: | ||||
|         result and creates a landmark with the associated details such as name, type, and OSM ID. | ||||
| 
 | ||||
|         Parameters: | ||||
|             shopping_location (ShoppingLocation): A ShoppingLocation object containing  | ||||
|             shopping_location (Cluster): A Cluster object containing  | ||||
|             the centroid and importance of the area. | ||||
| 
 | ||||
|         Returns: | ||||
| @@ -196,14 +187,21 @@ class ShoppingManager: | ||||
|         """ | ||||
| 
 | ||||
|         # Define the bounding box for a given radius around the coordinates | ||||
|         lat, lon = shopping_location.centroid | ||||
|         lat, lon = cluster.centroid | ||||
|         bbox = ("around:1000", str(lat), str(lon)) | ||||
| 
 | ||||
|         # Query neighborhoods and shopping malls | ||||
|         selectors = ['"place"~"^(suburb|neighborhood|neighbourhood|quarter|city_block)$"', '"shop"="mall"'] | ||||
|         selectors = ['"place"~"^(suburb|neighborhood|neighbourhood|quarter|city_block)$"'] | ||||
| 
 | ||||
|         if self.cluster_type == 'shopping' : | ||||
|             selectors.append('"shop"="mall"') | ||||
|             new_name = 'Shopping Area' | ||||
|             t = 40 | ||||
|         else :  | ||||
|             new_name = 'Neighborhood' | ||||
|             t = 15 | ||||
| 
 | ||||
|         min_dist = float('inf') | ||||
|         new_name = 'Shopping Area' | ||||
|         new_name_en = None | ||||
|         osm_id = 0 | ||||
|         osm_type = 'node' | ||||
| @@ -231,7 +229,7 @@ class ShoppingManager: | ||||
|                     if location[0] is None :  | ||||
|                         continue | ||||
| 
 | ||||
|                 d = get_distance(shopping_location.centroid, location) | ||||
|                 d = get_distance(cluster.centroid, location) | ||||
|                 if  d < min_dist : | ||||
|                     min_dist = d | ||||
|                     new_name = elem.tag('name') | ||||
| @@ -246,13 +244,14 @@ class ShoppingManager: | ||||
|          | ||||
|         return Landmark( | ||||
|             name=new_name, | ||||
|             type='shopping', | ||||
|             location=shopping_location.centroid,              # TODO: use the fact the we can also recognize streets. | ||||
|             attractiveness=shopping_location.importance, | ||||
|             type=self.cluster_type, | ||||
|             location=cluster.centroid,              # TODO: use the fact the we can also recognize streets. | ||||
|             attractiveness=cluster.importance, | ||||
|             n_tags=0, | ||||
|             osm_id=osm_id, | ||||
|             osm_type=osm_type, | ||||
|             name_en=new_name_en | ||||
|             name_en=new_name_en, | ||||
|             duration=t | ||||
|         ) | ||||
| 
 | ||||
| 
 | ||||
| @@ -5,7 +5,7 @@ from OSMPythonTools.cachingStrategy import CachingStrategy, JSON | ||||
| from ..structs.preferences import Preferences | ||||
| from ..structs.landmark import Landmark | ||||
| from .take_most_important import take_most_important | ||||
| from .cluster_processing import ShoppingManager | ||||
| from .cluster_manager import ClusterManager | ||||
|  | ||||
| from ..constants import AMENITY_SELECTORS_PATH, LANDMARK_PARAMETERS_PATH, OPTIMIZER_PARAMETERS_PATH, OSM_CACHE_DIR | ||||
|  | ||||
| @@ -86,6 +86,11 @@ class LandmarkManager: | ||||
|             current_landmarks = self.fetch_landmarks(bbox, self.amenity_selectors['sightseeing'], preferences.sightseeing.type, score_function) | ||||
|             all_landmarks.update(current_landmarks) | ||||
|  | ||||
|             # special pipeline for historic neighborhoods | ||||
|             neighborhood_manager = ClusterManager(bbox, 'sightseeing') | ||||
|             historic_clusters = neighborhood_manager.generate_clusters() | ||||
|             all_landmarks.update(historic_clusters) | ||||
|  | ||||
|         # list for nature | ||||
|         if preferences.nature.score != 0: | ||||
|             score_function = lambda score: score * 10 * self.nature_coeff * preferences.nature.score / 5 | ||||
| @@ -102,11 +107,9 @@ class LandmarkManager: | ||||
|             all_landmarks.update(current_landmarks) | ||||
|  | ||||
|             # special pipeline for shopping malls | ||||
|             shopping_manager = ShoppingManager(bbox) | ||||
|             if shopping_manager.valid : | ||||
|                 shopping_clusters = shopping_manager.generate_shopping_landmarks() | ||||
|                 for landmark in shopping_clusters : landmark.duration = 45 | ||||
|                 all_landmarks.update(shopping_clusters) | ||||
|             shopping_manager = ClusterManager(bbox, 'shopping') | ||||
|             shopping_clusters = shopping_manager.generate_clusters() | ||||
|             all_landmarks.update(shopping_clusters) | ||||
|              | ||||
|  | ||||
|  | ||||
| @@ -277,6 +280,11 @@ class LandmarkManager: | ||||
|                         skip = True | ||||
|                         break | ||||
|  | ||||
|                     if "building:" in tag_key: | ||||
|                         # do not count the building description as being particularly useful | ||||
|                         n_tags -= 1 | ||||
|                      | ||||
|  | ||||
|                     if "boundary" in tag_key: | ||||
|                         # skip "areas" like administrative boundaries and stuff | ||||
|                         skip = True | ||||
| @@ -327,13 +335,16 @@ class LandmarkManager: | ||||
|                     continue | ||||
|  | ||||
|                 score = score_function(score) | ||||
|                 if "place_of_worship" in elem.tags().values(): | ||||
|                     score = score * self.church_coeff | ||||
|                     duration = 10 | ||||
|                 if "place_of_worship" in elem.tags().values() : | ||||
|                     if "cathedral" not in elem.tags().values() : | ||||
|                         score = score * self.church_coeff | ||||
|                         duration = 5 | ||||
|                     else :  | ||||
|                         duration = 10 | ||||
|  | ||||
|                 if 'viewpoint' in elem.tags().values() : | ||||
|                 elif 'viewpoint' in elem.tags().values() : | ||||
|                     # viewpoints must count more | ||||
|                     score += self.viewpoint_bonus | ||||
|                     score = score * self.viewpoint_bonus | ||||
|                     duration = 10 | ||||
|                  | ||||
|                 elif "museum" in elem.tags().values() or "aquarium" in elem.tags().values() or "planetarium" in elem.tags().values(): | ||||
|   | ||||
| @@ -39,7 +39,7 @@ jobs: | ||||
|           # remove the 'v' prefix from the tag name | ||||
|           echo "BUILD_NAME=${REF_NAME//v}" >> $GITHUB_ENV | ||||
|  | ||||
|       - name: Load secrets from github | ||||
|       - name: Put selected secrets into files | ||||
|         run: | | ||||
|           echo "${{ secrets.ANDROID_SECRET_PROPERTIES_BASE64 }}" | base64 -d > secrets.properties | ||||
|           echo "${{ secrets.ANDROID_GOOGLE_PLAY_JSON_BASE64 }}" | base64 -d > google-key.json | ||||
| @@ -51,8 +51,9 @@ jobs: | ||||
|         working-directory: android | ||||
|  | ||||
|       - name: Run fastlane lane | ||||
|         run: bundle exec fastlane deploy_testing | ||||
|         run: bundle exec fastlane deploy_release | ||||
|         working-directory: android | ||||
|         env: | ||||
|           BUILD_NUMBER: ${{ github.run_number }} | ||||
|           # BUILD_NAME is implicitly available | ||||
|           GOOGLE_MAPS_API_KEY: ${{ secrets.GOOGLE_MAPS_API_KEY }} | ||||
|   | ||||
							
								
								
									
										64
									
								
								frontend/.github/workflows/build_app_ios.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								frontend/.github/workflows/build_app_ios.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| on: | ||||
|   push: | ||||
|     tags: | ||||
|       - 'v*' | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     runs-on: macos-latest | ||||
|     env: | ||||
|       # $BUNDLE_GEMFILE must be set at the job level, so it is set for all steps | ||||
|       BUNDLE_GEMFILE: ios/Gemfile | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Set up ruby env | ||||
|         uses: ruby/setup-ruby@v1 | ||||
|         with: | ||||
|           ruby-version: 3.3 | ||||
|           bundler-cache: true # runs 'bundle install' and caches installed gems automatically | ||||
|  | ||||
|       - name: Install Flutter | ||||
|         uses: subosito/flutter-action@v2 | ||||
|         with: | ||||
|           channel: stable | ||||
|           flutter-version: 3.22.0 | ||||
|           cache: true | ||||
|  | ||||
|       - name: Infer version number from git tag | ||||
|         id: version | ||||
|         env:  | ||||
|           REF_NAME: ${{ github.ref_name }} | ||||
|         run: | ||||
|           # remove the 'v' prefix from the tag name | ||||
|           echo "BUILD_NAME=${REF_NAME//v}" >> $GITHUB_ENV | ||||
|  | ||||
|       - name: Setup SSH key for match git repo | ||||
|         # and mark the host as known | ||||
|         run: | | ||||
|           echo $MATCH_REPO_SSH_KEY | base64 --decode > ~/.ssh/id_rsa | ||||
|           chmod 600 ~/.ssh/id_rsa | ||||
|           ssh-keyscan -p 2222 git.kluster.moll.re > ~/.ssh/known_hosts | ||||
|         env: | ||||
|           MATCH_REPO_SSH_KEY: ${{ secrets.IOS_MATCH_REPO_SSH_KEY_BASE64 }} | ||||
|  | ||||
|       - name: Install dependencies and clean up | ||||
|         run: | | ||||
|           flutter pub get | ||||
|           bundle exec pod install | ||||
|           flutter clean | ||||
|           bundle exec pod cache clean --all | ||||
|         working-directory: ios | ||||
|  | ||||
|       - name: Run fastlane lane | ||||
|         run: bundle exec fastlane deploy_release --verbose | ||||
|         working-directory: ios | ||||
|         env: | ||||
|           BUILD_NUMBER: ${{ github.run_number }} | ||||
|           # BUILD_NAME is implicitly available | ||||
|           GOOGLE_MAPS_API_KEY: ${{ secrets.GOOGLE_MAPS_API_KEY }} | ||||
|           IOS_ASC_KEY_ID: ${{ secrets.IOS_ASC_KEY_ID }} | ||||
|           IOS_ASC_ISSUER_ID: ${{ secrets.IOS_ASC_ISSUER_ID }} | ||||
|           IOS_ASC_KEY: ${{ secrets.IOS_ASC_KEY }} | ||||
|           MATCH_PASSWORD: ${{ secrets.IOS_MATCH_PASSWORD }} | ||||
|           IOS_GOOGLE_MAPS_API_KEY: ${{ secrets.IOS_GOOGLE_MAPS_API_KEY }} | ||||
| @@ -46,12 +46,16 @@ bundle exec fastlane <lane> | ||||
| ``` | ||||
| This is reused in the CI/CD pipeline to automate the deployment process. | ||||
|  | ||||
| Fastlane assumes mutliple secrets to be present as files in the platform directories. These are: | ||||
| - for android: | ||||
|     - `secrets.properties` used by gradle to load secrets needed at execution time | ||||
|     - `release.keystore` used by gradle to sign the apk | ||||
|     - `google-key.json` used by fastlane to authenticate with the Google Play Store | ||||
| - for ios: | ||||
|     - TODO | ||||
| Secrets used by fastlane are stored on hashicorp vault and are fetched by the CI/CD pipeline. See below. | ||||
|  | ||||
| These files are stored as secrets in the GitHub repository so that the CI pipeline can access them. | ||||
| ## Secrets | ||||
| These are mostly used by the CI/CD pipeline to deploy the application. The main usage for github actions is documented under [https://github.com/hashicorp/vault-action](https://github.com/hashicorp/vault-action). | ||||
|  | ||||
| **Platform-specific secrets** are used by the CI/CD pipeline to deploy to the respective app stores. | ||||
| - `GOOGLE_MAPS_API_KEY` is used to authenticate with the Google Maps API and is scoped to the android platform | ||||
| - `ANDROID_KEYSTORE` is used to sign the android apk | ||||
| - `ANDROID_GOOGLE_KEY` is used to authenticate with the Google Play Store api | ||||
| - `IOS_GOOGLE_MAPS_API_KEY` is used to authenticate with the Google Maps API and is scoped to the ios platform | ||||
| - `IOS_GOOGLE_...` | ||||
| - `IOS_GOOGLE_...` | ||||
| - `IOS_GOOGLE_...` | ||||
| @@ -63,11 +63,3 @@ Compared to the flutter template application, a few changes have to be made: | ||||
|         } | ||||
|     ``` | ||||
|  | ||||
|  | ||||
| ### Using the credentials in CI | ||||
| - Add the secret files to the repository secrets (e.g. `ANDROID_SECRETS_PROPERTIES`). | ||||
|  | ||||
| - temporarily write them back to files during the CI execution: | ||||
|     ```bash | ||||
|     echo {{ secrets.ANDROID_SECRETS }} >> android/secrets.properties | ||||
|     ``` | ||||
|   | ||||
| @@ -65,7 +65,7 @@ android { | ||||
|     } | ||||
|  | ||||
|     defaultConfig { | ||||
|         // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). | ||||
|    | ||||
|         applicationId "com.anydev.anyway" | ||||
|         // You can update the following values to match your application needs. | ||||
|         // For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration. | ||||
| @@ -77,7 +77,7 @@ android { | ||||
|         versionCode flutterVersionCode.toInteger() | ||||
|         versionName flutterVersionName | ||||
|         // // Placeholders of keys that are replaced by the build system. | ||||
|         manifestPlaceholders += ['MAPS_API_KEY': secretProperties.getProperty('MAPS_API_KEY')] | ||||
|         manifestPlaceholders += ['MAPS_API_KEY': System.getenv('GOOGLE_MAPS_API_KEY')] | ||||
|  | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,2 @@ | ||||
| # This file mirrors the state of secrets.properties as a reference for the developer. | ||||
| # And as a fallback for build.gradle | ||||
| MAPS_API_KEY=Key | ||||
| @@ -1,11 +1,8 @@ | ||||
| # Uncomment the line if you want fastlane to automatically update itself | ||||
| # update_fastlane | ||||
|  | ||||
| default_platform(:android) | ||||
|  | ||||
| platform :android do | ||||
|  | ||||
|   desc "Deploy a new version to closed testing" | ||||
|   desc "Deploy a new version to closed testing (play store)" | ||||
|   lane :deploy_testing do | ||||
|     build_name = ENV["BUILD_NAME"] | ||||
|     build_number = ENV["BUILD_NUMBER"] | ||||
| @@ -30,24 +27,26 @@ platform :android do | ||||
|       ) | ||||
|   end | ||||
|  | ||||
|  | ||||
|   desc "Deploy a new version as a full release" | ||||
|   lane :deploy_release do | ||||
|     gradle( | ||||
|       task: "clean assembleRelease", | ||||
|       # todo update to a flutter call | ||||
|       properties: { | ||||
|         # loaded from environment | ||||
|         "android.injected.version.name" => ENV["VERSION_NAME"], | ||||
|       } | ||||
|     build_name = ENV["BUILD_NAME"] | ||||
|     build_number = ENV["BUILD_NUMBER"] | ||||
|  | ||||
|     sh( | ||||
|       "flutter", | ||||
|       "build", | ||||
|       "appbundle", | ||||
|       "--release", | ||||
|       "--build-name=#{build_name}", | ||||
|       "--build-number=#{build_number}", | ||||
|       ) | ||||
|      | ||||
|     upload_to_play_store( | ||||
|       track: "production", | ||||
|       track: 'production', | ||||
|       skip_upload_apk: true, | ||||
|       skip_upload_changelogs: true, | ||||
|       aab: "../build/app/outputs/bundle/release/app-release.aab", | ||||
|       # this is the default output of flutter build ... --release | ||||
|       # in particular this the build folder lies in the flutter root folder | ||||
|       # this is the parent folder for the android folder | ||||
|     ) | ||||
|       ) | ||||
|   end | ||||
| end | ||||
|   | ||||
							
								
								
									
										6
									
								
								frontend/ios/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								frontend/ios/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +1,9 @@ | ||||
| # fastlane secret | ||||
| .env | ||||
| secret.env | ||||
| *.mobileprovision | ||||
| report.xml | ||||
|  | ||||
| **/dgph | ||||
| *.mode1v3 | ||||
| *.mode2v3 | ||||
|   | ||||
| @@ -1 +1,2 @@ | ||||
| #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig" | ||||
| #include "Generated.xcconfig" | ||||
|   | ||||
| @@ -1 +1,2 @@ | ||||
| #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig" | ||||
| #include "Generated.xcconfig" | ||||
|   | ||||
							
								
								
									
										5
									
								
								frontend/ios/Gemfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								frontend/ios/Gemfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| source "https://rubygems.org" | ||||
|  | ||||
| gem "fastlane" | ||||
| gem "cocoapods" | ||||
|  | ||||
							
								
								
									
										288
									
								
								frontend/ios/Gemfile.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										288
									
								
								frontend/ios/Gemfile.lock
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,288 @@ | ||||
| GEM | ||||
|   remote: https://rubygems.org/ | ||||
|   specs: | ||||
|     CFPropertyList (3.0.7) | ||||
|       base64 | ||||
|       nkf | ||||
|       rexml | ||||
|     activesupport (5.2.8.1) | ||||
|       concurrent-ruby (~> 1.0, >= 1.0.2) | ||||
|       i18n (>= 0.7, < 2) | ||||
|       minitest (~> 5.1) | ||||
|       tzinfo (~> 1.1) | ||||
|     addressable (2.8.7) | ||||
|       public_suffix (>= 2.0.2, < 7.0) | ||||
|     algoliasearch (1.27.5) | ||||
|       httpclient (~> 2.8, >= 2.8.3) | ||||
|       json (>= 1.5.1) | ||||
|     artifactory (3.0.17) | ||||
|     atomos (0.1.3) | ||||
|     aws-eventstream (1.3.0) | ||||
|     aws-partitions (1.1004.0) | ||||
|     aws-sdk-core (3.212.0) | ||||
|       aws-eventstream (~> 1, >= 1.3.0) | ||||
|       aws-partitions (~> 1, >= 1.992.0) | ||||
|       aws-sigv4 (~> 1.9) | ||||
|       jmespath (~> 1, >= 1.6.1) | ||||
|     aws-sdk-kms (1.95.0) | ||||
|       aws-sdk-core (~> 3, >= 3.210.0) | ||||
|       aws-sigv4 (~> 1.5) | ||||
|     aws-sdk-s3 (1.170.1) | ||||
|       aws-sdk-core (~> 3, >= 3.210.0) | ||||
|       aws-sdk-kms (~> 1) | ||||
|       aws-sigv4 (~> 1.5) | ||||
|     aws-sigv4 (1.10.1) | ||||
|       aws-eventstream (~> 1, >= 1.0.2) | ||||
|     babosa (1.0.4) | ||||
|     base64 (0.2.0) | ||||
|     claide (1.1.0) | ||||
|     cocoapods (1.10.2) | ||||
|       addressable (~> 2.6) | ||||
|       claide (>= 1.0.2, < 2.0) | ||||
|       cocoapods-core (= 1.10.2) | ||||
|       cocoapods-deintegrate (>= 1.0.3, < 2.0) | ||||
|       cocoapods-downloader (>= 1.4.0, < 2.0) | ||||
|       cocoapods-plugins (>= 1.0.0, < 2.0) | ||||
|       cocoapods-search (>= 1.0.0, < 2.0) | ||||
|       cocoapods-trunk (>= 1.4.0, < 2.0) | ||||
|       cocoapods-try (>= 1.1.0, < 2.0) | ||||
|       colored2 (~> 3.1) | ||||
|       escape (~> 0.0.4) | ||||
|       fourflusher (>= 2.3.0, < 3.0) | ||||
|       gh_inspector (~> 1.0) | ||||
|       molinillo (~> 0.6.6) | ||||
|       nap (~> 1.0) | ||||
|       ruby-macho (~> 1.4) | ||||
|       xcodeproj (>= 1.19.0, < 2.0) | ||||
|     cocoapods-core (1.10.2) | ||||
|       activesupport (> 5.0, < 6) | ||||
|       addressable (~> 2.6) | ||||
|       algoliasearch (~> 1.0) | ||||
|       concurrent-ruby (~> 1.1) | ||||
|       fuzzy_match (~> 2.0.4) | ||||
|       nap (~> 1.0) | ||||
|       netrc (~> 0.11) | ||||
|       public_suffix | ||||
|       typhoeus (~> 1.0) | ||||
|     cocoapods-deintegrate (1.0.5) | ||||
|     cocoapods-downloader (1.6.3) | ||||
|     cocoapods-plugins (1.0.0) | ||||
|       nap | ||||
|     cocoapods-search (1.0.1) | ||||
|     cocoapods-trunk (1.6.0) | ||||
|       nap (>= 0.8, < 2.0) | ||||
|       netrc (~> 0.11) | ||||
|     cocoapods-try (1.2.0) | ||||
|     colored (1.2) | ||||
|     colored2 (3.1.2) | ||||
|     commander (4.6.0) | ||||
|       highline (~> 2.0.0) | ||||
|     concurrent-ruby (1.3.4) | ||||
|     declarative (0.0.20) | ||||
|     digest-crc (0.6.5) | ||||
|       rake (>= 12.0.0, < 14.0.0) | ||||
|     domain_name (0.6.20240107) | ||||
|     dotenv (2.8.1) | ||||
|     emoji_regex (3.2.3) | ||||
|     escape (0.0.4) | ||||
|     ethon (0.16.0) | ||||
|       ffi (>= 1.15.0) | ||||
|     excon (0.112.0) | ||||
|     faraday (1.10.4) | ||||
|       faraday-em_http (~> 1.0) | ||||
|       faraday-em_synchrony (~> 1.0) | ||||
|       faraday-excon (~> 1.1) | ||||
|       faraday-httpclient (~> 1.0) | ||||
|       faraday-multipart (~> 1.0) | ||||
|       faraday-net_http (~> 1.0) | ||||
|       faraday-net_http_persistent (~> 1.0) | ||||
|       faraday-patron (~> 1.0) | ||||
|       faraday-rack (~> 1.0) | ||||
|       faraday-retry (~> 1.0) | ||||
|       ruby2_keywords (>= 0.0.4) | ||||
|     faraday-cookie_jar (0.0.7) | ||||
|       faraday (>= 0.8.0) | ||||
|       http-cookie (~> 1.0.0) | ||||
|     faraday-em_http (1.0.0) | ||||
|     faraday-em_synchrony (1.0.0) | ||||
|     faraday-excon (1.1.0) | ||||
|     faraday-httpclient (1.0.1) | ||||
|     faraday-multipart (1.0.4) | ||||
|       multipart-post (~> 2) | ||||
|     faraday-net_http (1.0.2) | ||||
|     faraday-net_http_persistent (1.2.0) | ||||
|     faraday-patron (1.0.0) | ||||
|     faraday-rack (1.0.0) | ||||
|     faraday-retry (1.0.3) | ||||
|     faraday_middleware (1.2.1) | ||||
|       faraday (~> 1.0) | ||||
|     fastimage (2.3.1) | ||||
|     fastlane (2.225.0) | ||||
|       CFPropertyList (>= 2.3, < 4.0.0) | ||||
|       addressable (>= 2.8, < 3.0.0) | ||||
|       artifactory (~> 3.0) | ||||
|       aws-sdk-s3 (~> 1.0) | ||||
|       babosa (>= 1.0.3, < 2.0.0) | ||||
|       bundler (>= 1.12.0, < 3.0.0) | ||||
|       colored (~> 1.2) | ||||
|       commander (~> 4.6) | ||||
|       dotenv (>= 2.1.1, < 3.0.0) | ||||
|       emoji_regex (>= 0.1, < 4.0) | ||||
|       excon (>= 0.71.0, < 1.0.0) | ||||
|       faraday (~> 1.0) | ||||
|       faraday-cookie_jar (~> 0.0.6) | ||||
|       faraday_middleware (~> 1.0) | ||||
|       fastimage (>= 2.1.0, < 3.0.0) | ||||
|       fastlane-sirp (>= 1.0.0) | ||||
|       gh_inspector (>= 1.1.2, < 2.0.0) | ||||
|       google-apis-androidpublisher_v3 (~> 0.3) | ||||
|       google-apis-playcustomapp_v1 (~> 0.1) | ||||
|       google-cloud-env (>= 1.6.0, < 2.0.0) | ||||
|       google-cloud-storage (~> 1.31) | ||||
|       highline (~> 2.0) | ||||
|       http-cookie (~> 1.0.5) | ||||
|       json (< 3.0.0) | ||||
|       jwt (>= 2.1.0, < 3) | ||||
|       mini_magick (>= 4.9.4, < 5.0.0) | ||||
|       multipart-post (>= 2.0.0, < 3.0.0) | ||||
|       naturally (~> 2.2) | ||||
|       optparse (>= 0.1.1, < 1.0.0) | ||||
|       plist (>= 3.1.0, < 4.0.0) | ||||
|       rubyzip (>= 2.0.0, < 3.0.0) | ||||
|       security (= 0.1.5) | ||||
|       simctl (~> 1.6.3) | ||||
|       terminal-notifier (>= 2.0.0, < 3.0.0) | ||||
|       terminal-table (~> 3) | ||||
|       tty-screen (>= 0.6.3, < 1.0.0) | ||||
|       tty-spinner (>= 0.8.0, < 1.0.0) | ||||
|       word_wrap (~> 1.0.0) | ||||
|       xcodeproj (>= 1.13.0, < 2.0.0) | ||||
|       xcpretty (~> 0.3.0) | ||||
|       xcpretty-travis-formatter (>= 0.0.3, < 2.0.0) | ||||
|     fastlane-sirp (1.0.0) | ||||
|       sysrandom (~> 1.0) | ||||
|     ffi (1.17.0) | ||||
|     ffi (1.17.0-x86_64-darwin) | ||||
|     fourflusher (2.3.1) | ||||
|     fuzzy_match (2.0.4) | ||||
|     gh_inspector (1.1.3) | ||||
|     google-apis-androidpublisher_v3 (0.54.0) | ||||
|       google-apis-core (>= 0.11.0, < 2.a) | ||||
|     google-apis-core (0.11.3) | ||||
|       addressable (~> 2.5, >= 2.5.1) | ||||
|       googleauth (>= 0.16.2, < 2.a) | ||||
|       httpclient (>= 2.8.1, < 3.a) | ||||
|       mini_mime (~> 1.0) | ||||
|       representable (~> 3.0) | ||||
|       retriable (>= 2.0, < 4.a) | ||||
|       rexml | ||||
|     google-apis-iamcredentials_v1 (0.17.0) | ||||
|       google-apis-core (>= 0.11.0, < 2.a) | ||||
|     google-apis-playcustomapp_v1 (0.13.0) | ||||
|       google-apis-core (>= 0.11.0, < 2.a) | ||||
|     google-apis-storage_v1 (0.31.0) | ||||
|       google-apis-core (>= 0.11.0, < 2.a) | ||||
|     google-cloud-core (1.7.1) | ||||
|       google-cloud-env (>= 1.0, < 3.a) | ||||
|       google-cloud-errors (~> 1.0) | ||||
|     google-cloud-env (1.6.0) | ||||
|       faraday (>= 0.17.3, < 3.0) | ||||
|     google-cloud-errors (1.4.0) | ||||
|     google-cloud-storage (1.47.0) | ||||
|       addressable (~> 2.8) | ||||
|       digest-crc (~> 0.4) | ||||
|       google-apis-iamcredentials_v1 (~> 0.1) | ||||
|       google-apis-storage_v1 (~> 0.31.0) | ||||
|       google-cloud-core (~> 1.6) | ||||
|       googleauth (>= 0.16.2, < 2.a) | ||||
|       mini_mime (~> 1.0) | ||||
|     googleauth (1.8.1) | ||||
|       faraday (>= 0.17.3, < 3.a) | ||||
|       jwt (>= 1.4, < 3.0) | ||||
|       multi_json (~> 1.11) | ||||
|       os (>= 0.9, < 2.0) | ||||
|       signet (>= 0.16, < 2.a) | ||||
|     highline (2.0.3) | ||||
|     http-cookie (1.0.7) | ||||
|       domain_name (~> 0.5) | ||||
|     httpclient (2.8.3) | ||||
|     i18n (1.14.6) | ||||
|       concurrent-ruby (~> 1.0) | ||||
|     jmespath (1.6.2) | ||||
|     json (2.8.1) | ||||
|     jwt (2.9.3) | ||||
|       base64 | ||||
|     mini_magick (4.13.2) | ||||
|     mini_mime (1.1.5) | ||||
|     minitest (5.25.1) | ||||
|     molinillo (0.6.6) | ||||
|     multi_json (1.15.0) | ||||
|     multipart-post (2.4.1) | ||||
|     nanaimo (0.4.0) | ||||
|     nap (1.1.0) | ||||
|     naturally (2.2.1) | ||||
|     netrc (0.11.0) | ||||
|     nkf (0.2.0) | ||||
|     optparse (0.6.0) | ||||
|     os (1.1.4) | ||||
|     plist (3.7.1) | ||||
|     public_suffix (6.0.1) | ||||
|     rake (13.2.1) | ||||
|     representable (3.2.0) | ||||
|       declarative (< 0.1.0) | ||||
|       trailblazer-option (>= 0.1.1, < 0.2.0) | ||||
|       uber (< 0.2.0) | ||||
|     retriable (3.1.2) | ||||
|     rexml (3.3.9) | ||||
|     rouge (2.0.7) | ||||
|     ruby-macho (1.4.0) | ||||
|     ruby2_keywords (0.0.5) | ||||
|     rubyzip (2.3.2) | ||||
|     security (0.1.5) | ||||
|     signet (0.19.0) | ||||
|       addressable (~> 2.8) | ||||
|       faraday (>= 0.17.5, < 3.a) | ||||
|       jwt (>= 1.5, < 3.0) | ||||
|       multi_json (~> 1.10) | ||||
|     simctl (1.6.10) | ||||
|       CFPropertyList | ||||
|       naturally | ||||
|     sysrandom (1.0.5) | ||||
|     terminal-notifier (2.0.0) | ||||
|     terminal-table (3.0.2) | ||||
|       unicode-display_width (>= 1.1.1, < 3) | ||||
|     thread_safe (0.3.6) | ||||
|     trailblazer-option (0.1.2) | ||||
|     tty-cursor (0.7.1) | ||||
|     tty-screen (0.8.2) | ||||
|     tty-spinner (0.9.3) | ||||
|       tty-cursor (~> 0.7) | ||||
|     typhoeus (1.4.1) | ||||
|       ethon (>= 0.9.0) | ||||
|     tzinfo (1.2.11) | ||||
|       thread_safe (~> 0.1) | ||||
|     uber (0.1.0) | ||||
|     unicode-display_width (2.6.0) | ||||
|     word_wrap (1.0.0) | ||||
|     xcodeproj (1.27.0) | ||||
|       CFPropertyList (>= 2.3.3, < 4.0) | ||||
|       atomos (~> 0.1.3) | ||||
|       claide (>= 1.0.2, < 2.0) | ||||
|       colored2 (~> 3.1) | ||||
|       nanaimo (~> 0.4.0) | ||||
|       rexml (>= 3.3.6, < 4.0) | ||||
|     xcpretty (0.3.0) | ||||
|       rouge (~> 2.0.7) | ||||
|     xcpretty-travis-formatter (1.0.1) | ||||
|       xcpretty (~> 0.2, >= 0.0.7) | ||||
|  | ||||
| PLATFORMS | ||||
|   ruby | ||||
|   x86_64-darwin-23 | ||||
|  | ||||
| DEPENDENCIES | ||||
|   cocoapods | ||||
|   fastlane | ||||
|  | ||||
| BUNDLED WITH | ||||
|    2.5.23 | ||||
							
								
								
									
										59
									
								
								frontend/ios/Podfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								frontend/ios/Podfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| # Uncomment this line to define a global platform for your project | ||||
| # platform :ios, '12.0' | ||||
|  | ||||
| # CocoaPods analytics sends network stats synchronously affecting flutter build latency. | ||||
| ENV['COCOAPODS_DISABLE_STATS'] = 'true' | ||||
|  | ||||
| project 'Runner', { | ||||
|   'Debug' => :debug, | ||||
|   'Profile' => :release, | ||||
|   'Release' => :release, | ||||
| } | ||||
|  | ||||
| def flutter_root | ||||
|   generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'Generated.xcconfig'), __FILE__) | ||||
|   unless File.exist?(generated_xcode_build_settings_path) | ||||
|     raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure flutter pub get is executed first" | ||||
|   end | ||||
|  | ||||
|   File.foreach(generated_xcode_build_settings_path) do |line| | ||||
|     matches = line.match(/FLUTTER_ROOT\=(.*)/) | ||||
|     return matches[1].strip if matches | ||||
|   end | ||||
|   raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Generated.xcconfig, then run flutter pub get" | ||||
| end | ||||
|  | ||||
| require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root) | ||||
|  | ||||
| flutter_ios_podfile_setup | ||||
|  | ||||
| target 'Runner' do | ||||
|   use_frameworks! | ||||
|   use_modular_headers! | ||||
|  | ||||
|   flutter_install_all_ios_pods File.dirname(File.realpath(__FILE__)) | ||||
|   target 'RunnerTests' do | ||||
|     inherit! :search_paths | ||||
|   end | ||||
| end | ||||
|  | ||||
| post_install do |installer| | ||||
|   installer.pods_project.targets.each do |target| | ||||
|     flutter_additional_ios_build_settings(target) | ||||
|  | ||||
|     target.build_configurations.each do |config| | ||||
|       # You can remove unused permissions here | ||||
|       # for more information: https://github.com/BaseflowIT/flutter-permission-handler/blob/master/permission_handler/ios/Classes/PermissionHandlerEnums.h | ||||
|       config.build_settings['GCC_PREPROCESSOR_DEFINITIONS'] ||= [ | ||||
|         '$(inherited)', | ||||
|         ## The 'PERMISSION_LOCATION' macro enables the `locationWhenInUse` and `locationAlways` permission. If | ||||
|         ## the application only requires `locationWhenInUse`, only specify the `PERMISSION_LOCATION_WHENINUSE` | ||||
|         ## macro. | ||||
|         ## | ||||
|         ## dart: [PermissionGroup.location, PermissionGroup.locationAlways, PermissionGroup.locationWhenInUse] | ||||
|         'PERMISSION_LOCATION=1', | ||||
|         'PERMISSION_LOCATION_WHENINUSE=0', | ||||
|       ] | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										87
									
								
								frontend/ios/Podfile.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								frontend/ios/Podfile.lock
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| PODS: | ||||
|   - Flutter (1.0.0) | ||||
|   - geocoding_ios (1.0.5): | ||||
|     - Flutter | ||||
|   - geolocator_apple (1.2.0): | ||||
|     - Flutter | ||||
|   - Google-Maps-iOS-Utils (6.1.0): | ||||
|     - GoogleMaps (~> 9.0) | ||||
|   - google_maps_flutter_ios (0.0.1): | ||||
|     - Flutter | ||||
|     - Google-Maps-iOS-Utils (< 7.0, >= 5.0) | ||||
|     - GoogleMaps (< 10.0, >= 8.4) | ||||
|   - GoogleMaps (9.2.0): | ||||
|     - GoogleMaps/Maps (= 9.2.0) | ||||
|   - GoogleMaps/Maps (9.2.0) | ||||
|   - map_launcher (0.0.1): | ||||
|     - Flutter | ||||
|   - path_provider_foundation (0.0.1): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|   - permission_handler_apple (9.3.0): | ||||
|     - Flutter | ||||
|   - shared_preferences_foundation (0.0.1): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|   - sqflite (0.0.3): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|   - url_launcher_ios (0.0.1): | ||||
|     - Flutter | ||||
|  | ||||
| DEPENDENCIES: | ||||
|   - Flutter (from `Flutter`) | ||||
|   - geocoding_ios (from `.symlinks/plugins/geocoding_ios/ios`) | ||||
|   - geolocator_apple (from `.symlinks/plugins/geolocator_apple/ios`) | ||||
|   - google_maps_flutter_ios (from `.symlinks/plugins/google_maps_flutter_ios/ios`) | ||||
|   - map_launcher (from `.symlinks/plugins/map_launcher/ios`) | ||||
|   - path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`) | ||||
|   - permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`) | ||||
|   - shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`) | ||||
|   - sqflite (from `.symlinks/plugins/sqflite/darwin`) | ||||
|   - url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`) | ||||
|  | ||||
| SPEC REPOS: | ||||
|   trunk: | ||||
|     - Google-Maps-iOS-Utils | ||||
|     - GoogleMaps | ||||
|  | ||||
| EXTERNAL SOURCES: | ||||
|   Flutter: | ||||
|     :path: Flutter | ||||
|   geocoding_ios: | ||||
|     :path: ".symlinks/plugins/geocoding_ios/ios" | ||||
|   geolocator_apple: | ||||
|     :path: ".symlinks/plugins/geolocator_apple/ios" | ||||
|   google_maps_flutter_ios: | ||||
|     :path: ".symlinks/plugins/google_maps_flutter_ios/ios" | ||||
|   map_launcher: | ||||
|     :path: ".symlinks/plugins/map_launcher/ios" | ||||
|   path_provider_foundation: | ||||
|     :path: ".symlinks/plugins/path_provider_foundation/darwin" | ||||
|   permission_handler_apple: | ||||
|     :path: ".symlinks/plugins/permission_handler_apple/ios" | ||||
|   shared_preferences_foundation: | ||||
|     :path: ".symlinks/plugins/shared_preferences_foundation/darwin" | ||||
|   sqflite: | ||||
|     :path: ".symlinks/plugins/sqflite/darwin" | ||||
|   url_launcher_ios: | ||||
|     :path: ".symlinks/plugins/url_launcher_ios/ios" | ||||
|  | ||||
| SPEC CHECKSUMS: | ||||
|   Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 | ||||
|   geocoding_ios: bcbdaa6bddd7d3129c9bcb8acddc5d8778689768 | ||||
|   geolocator_apple: d981750b9f47dbdb02427e1476d9a04397beb8d9 | ||||
|   Google-Maps-iOS-Utils: 0a484b05ed21d88c9f9ebbacb007956edd508a96 | ||||
|   google_maps_flutter_ios: 0291eb2aa252298a769b04d075e4a9d747ff7264 | ||||
|   GoogleMaps: 634ec3ca99698b31ca2253d64f017217d70cfb38 | ||||
|   map_launcher: fe43bda6720bb73c12fcc1bdd86123ff49a4d4d6 | ||||
|   path_provider_foundation: 080d55be775b7414fd5a5ef3ac137b97b097e564 | ||||
|   permission_handler_apple: 4ed2196e43d0651e8ff7ca3483a069d469701f2d | ||||
|   shared_preferences_foundation: 9e1978ff2562383bd5676f64ec4e9aa8fa06a6f7 | ||||
|   sqflite: c35dad70033b8862124f8337cc994a809fcd9fa3 | ||||
|   url_launcher_ios: 694010445543906933d732453a59da0a173ae33d | ||||
|  | ||||
| PODFILE CHECKSUM: bd1a78910c05ac1e3a220e80f392c61ab2cc8789 | ||||
|  | ||||
| COCOAPODS: 1.10.2 | ||||
| @@ -11,9 +11,11 @@ | ||||
| 		331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C807B294A618700263BE5 /* RunnerTests.swift */; }; | ||||
| 		3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; }; | ||||
| 		74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74858FAE1ED2DC5600515810 /* AppDelegate.swift */; }; | ||||
| 		8F724AF5AC92A8A68D89C67E /* Pods_Runner.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 03CCEF89D4BD42ADA86AEDF9 /* Pods_Runner.framework */; }; | ||||
| 		97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; }; | ||||
| 		97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; }; | ||||
| 		97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; }; | ||||
| 		CDD1C9EB82AEC89C2181F722 /* Pods_RunnerTests.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4CB8B4133CEB7949B7EEBD81 /* Pods_RunnerTests.framework */; }; | ||||
| /* End PBXBuildFile section */ | ||||
|  | ||||
| /* Begin PBXContainerItemProxy section */ | ||||
| @@ -40,14 +42,20 @@ | ||||
| /* End PBXCopyFilesBuildPhase section */ | ||||
|  | ||||
| /* Begin PBXFileReference section */ | ||||
| 		03CCEF89D4BD42ADA86AEDF9 /* Pods_Runner.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_Runner.framework; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = "<group>"; }; | ||||
| 		1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = "<group>"; }; | ||||
| 		282EA28E78AB3F765E4BA719 /* Pods-RunnerTests.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.profile.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.profile.xcconfig"; sourceTree = "<group>"; }; | ||||
| 		3023467726A2A8275ED51C3E /* Pods-Runner.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.debug.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"; sourceTree = "<group>"; }; | ||||
| 		331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = "<group>"; }; | ||||
| 		331C8081294A63A400263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = "<group>"; }; | ||||
| 		4CB8B4133CEB7949B7EEBD81 /* Pods_RunnerTests.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_RunnerTests.framework; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		5F8BB7E700693DEAB89BBE69 /* Pods-Runner.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.release.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"; sourceTree = "<group>"; }; | ||||
| 		74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = "<group>"; }; | ||||
| 		74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; }; | ||||
| 		7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = "<group>"; }; | ||||
| 		7B8A81C772249160491754F9 /* Pods-Runner.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.profile.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.profile.xcconfig"; sourceTree = "<group>"; }; | ||||
| 		9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = "<group>"; }; | ||||
| 		9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = "<group>"; }; | ||||
| 		97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| @@ -55,19 +63,43 @@ | ||||
| 		97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }; | ||||
| 		97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = "<group>"; }; | ||||
| 		97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }; | ||||
| 		A565AAB9FE158487ABF3A5BF /* Pods-RunnerTests.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.release.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.release.xcconfig"; sourceTree = "<group>"; }; | ||||
| 		DC475F5210027479529644C3 /* Pods-RunnerTests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.debug.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.debug.xcconfig"; sourceTree = "<group>"; }; | ||||
| /* End PBXFileReference section */ | ||||
|  | ||||
| /* Begin PBXFrameworksBuildPhase section */ | ||||
| 		03EC59CC2AABC9D86B4ABFD7 /* Frameworks */ = { | ||||
| 			isa = PBXFrameworksBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 				CDD1C9EB82AEC89C2181F722 /* Pods_RunnerTests.framework in Frameworks */, | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| 		97C146EB1CF9000F007C117D /* Frameworks */ = { | ||||
| 			isa = PBXFrameworksBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 				8F724AF5AC92A8A68D89C67E /* Pods_Runner.framework in Frameworks */, | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| /* End PBXFrameworksBuildPhase section */ | ||||
|  | ||||
| /* Begin PBXGroup section */ | ||||
| 		1C946B8D83A95663C2489C91 /* Pods */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				3023467726A2A8275ED51C3E /* Pods-Runner.debug.xcconfig */, | ||||
| 				5F8BB7E700693DEAB89BBE69 /* Pods-Runner.release.xcconfig */, | ||||
| 				7B8A81C772249160491754F9 /* Pods-Runner.profile.xcconfig */, | ||||
| 				DC475F5210027479529644C3 /* Pods-RunnerTests.debug.xcconfig */, | ||||
| 				A565AAB9FE158487ABF3A5BF /* Pods-RunnerTests.release.xcconfig */, | ||||
| 				282EA28E78AB3F765E4BA719 /* Pods-RunnerTests.profile.xcconfig */, | ||||
| 			); | ||||
| 			path = Pods; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		331C8082294A63A400263BE5 /* RunnerTests */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| @@ -76,6 +108,15 @@ | ||||
| 			path = RunnerTests; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		3ECCC9BD7D0792871219624C /* Frameworks */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| 				03CCEF89D4BD42ADA86AEDF9 /* Pods_Runner.framework */, | ||||
| 				4CB8B4133CEB7949B7EEBD81 /* Pods_RunnerTests.framework */, | ||||
| 			); | ||||
| 			name = Frameworks; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		9740EEB11CF90186004384FC /* Flutter */ = { | ||||
| 			isa = PBXGroup; | ||||
| 			children = ( | ||||
| @@ -94,6 +135,8 @@ | ||||
| 				97C146F01CF9000F007C117D /* Runner */, | ||||
| 				97C146EF1CF9000F007C117D /* Products */, | ||||
| 				331C8082294A63A400263BE5 /* RunnerTests */, | ||||
| 				1C946B8D83A95663C2489C91 /* Pods */, | ||||
| 				3ECCC9BD7D0792871219624C /* Frameworks */, | ||||
| 			); | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| @@ -128,8 +171,10 @@ | ||||
| 			isa = PBXNativeTarget; | ||||
| 			buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */; | ||||
| 			buildPhases = ( | ||||
| 				F27C1B361CA1B045C8D36B3B /* [CP] Check Pods Manifest.lock */, | ||||
| 				331C807D294A63A400263BE5 /* Sources */, | ||||
| 				331C807F294A63A400263BE5 /* Resources */, | ||||
| 				03EC59CC2AABC9D86B4ABFD7 /* Frameworks */, | ||||
| 			); | ||||
| 			buildRules = ( | ||||
| 			); | ||||
| @@ -145,12 +190,15 @@ | ||||
| 			isa = PBXNativeTarget; | ||||
| 			buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */; | ||||
| 			buildPhases = ( | ||||
| 				2116AEE9DABFBBDED304ABEB /* [CP] Check Pods Manifest.lock */, | ||||
| 				9740EEB61CF901F6004384FC /* Run Script */, | ||||
| 				97C146EA1CF9000F007C117D /* Sources */, | ||||
| 				97C146EB1CF9000F007C117D /* Frameworks */, | ||||
| 				97C146EC1CF9000F007C117D /* Resources */, | ||||
| 				9705A1C41CF9048500538489 /* Embed Frameworks */, | ||||
| 				3B06AD1E1E4923F5004D2608 /* Thin Binary */, | ||||
| 				FE4BAF74959AF0624BA808EE /* [CP] Embed Pods Frameworks */, | ||||
| 				EE58653D94051600FD646EBE /* [CP] Copy Pods Resources */, | ||||
| 			); | ||||
| 			buildRules = ( | ||||
| 			); | ||||
| @@ -222,6 +270,28 @@ | ||||
| /* End PBXResourcesBuildPhase section */ | ||||
|  | ||||
| /* Begin PBXShellScriptBuildPhase section */ | ||||
| 		2116AEE9DABFBBDED304ABEB /* [CP] Check Pods Manifest.lock */ = { | ||||
| 			isa = PBXShellScriptBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			inputFileListPaths = ( | ||||
| 			); | ||||
| 			inputPaths = ( | ||||
| 				"${PODS_PODFILE_DIR_PATH}/Podfile.lock", | ||||
| 				"${PODS_ROOT}/Manifest.lock", | ||||
| 			); | ||||
| 			name = "[CP] Check Pods Manifest.lock"; | ||||
| 			outputFileListPaths = ( | ||||
| 			); | ||||
| 			outputPaths = ( | ||||
| 				"$(DERIVED_FILE_DIR)/Pods-Runner-checkManifestLockResult.txt", | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 			shellPath = /bin/sh; | ||||
| 			shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n    # print error to STDERR\n    echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n    exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n"; | ||||
| 			showEnvVarsInLog = 0; | ||||
| 		}; | ||||
| 		3B06AD1E1E4923F5004D2608 /* Thin Binary */ = { | ||||
| 			isa = PBXShellScriptBuildPhase; | ||||
| 			alwaysOutOfDate = 1; | ||||
| @@ -253,6 +323,62 @@ | ||||
| 			shellPath = /bin/sh; | ||||
| 			shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build"; | ||||
| 		}; | ||||
| 		EE58653D94051600FD646EBE /* [CP] Copy Pods Resources */ = { | ||||
| 			isa = PBXShellScriptBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			inputFileListPaths = ( | ||||
| 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-input-files.xcfilelist", | ||||
| 			); | ||||
| 			name = "[CP] Copy Pods Resources"; | ||||
| 			outputFileListPaths = ( | ||||
| 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-output-files.xcfilelist", | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 			shellPath = /bin/sh; | ||||
| 			shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources.sh\"\n"; | ||||
| 			showEnvVarsInLog = 0; | ||||
| 		}; | ||||
| 		F27C1B361CA1B045C8D36B3B /* [CP] Check Pods Manifest.lock */ = { | ||||
| 			isa = PBXShellScriptBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			inputFileListPaths = ( | ||||
| 			); | ||||
| 			inputPaths = ( | ||||
| 				"${PODS_PODFILE_DIR_PATH}/Podfile.lock", | ||||
| 				"${PODS_ROOT}/Manifest.lock", | ||||
| 			); | ||||
| 			name = "[CP] Check Pods Manifest.lock"; | ||||
| 			outputFileListPaths = ( | ||||
| 			); | ||||
| 			outputPaths = ( | ||||
| 				"$(DERIVED_FILE_DIR)/Pods-RunnerTests-checkManifestLockResult.txt", | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 			shellPath = /bin/sh; | ||||
| 			shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n    # print error to STDERR\n    echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n    exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n"; | ||||
| 			showEnvVarsInLog = 0; | ||||
| 		}; | ||||
| 		FE4BAF74959AF0624BA808EE /* [CP] Embed Pods Frameworks */ = { | ||||
| 			isa = PBXShellScriptBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			inputFileListPaths = ( | ||||
| 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-input-files.xcfilelist", | ||||
| 			); | ||||
| 			name = "[CP] Embed Pods Frameworks"; | ||||
| 			outputFileListPaths = ( | ||||
| 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-output-files.xcfilelist", | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 			shellPath = /bin/sh; | ||||
| 			shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh\"\n"; | ||||
| 			showEnvVarsInLog = 0; | ||||
| 		}; | ||||
| /* End PBXShellScriptBuildPhase section */ | ||||
|  | ||||
| /* Begin PBXSourcesBuildPhase section */ | ||||
| @@ -327,6 +453,7 @@ | ||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
| @@ -361,27 +488,45 @@ | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
| 				CLANG_ENABLE_MODULES = YES; | ||||
| 				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; | ||||
| 				CODE_SIGN_IDENTITY = "Apple Development"; | ||||
| 				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; | ||||
| 				CODE_SIGN_STYLE = Manual; | ||||
| 				CURRENT_PROJECT_VERSION = 3; | ||||
| 				DEVELOPMENT_TEAM = ""; | ||||
| 				"DEVELOPMENT_TEAM[sdk=iphoneos*]" = L32Y3D8V83; | ||||
| 				ENABLE_BITCODE = NO; | ||||
| 				INFOPLIST_FILE = Runner/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = Any.Way; | ||||
| 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel"; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.6; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 				); | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation; | ||||
| 				MARKETING_VERSION = 1.0.0; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = info.anydev.anyway; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				PROVISIONING_PROFILE_SPECIFIER = "match AppStore info.anydev.anyway"; | ||||
| 				"PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore info.anydev.anyway"; | ||||
| 				SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; | ||||
| 				SUPPORTS_MACCATALYST = NO; | ||||
| 				SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||
| 				SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||
| 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 				VERSIONING_SYSTEM = "apple-generic"; | ||||
| 			}; | ||||
| 			name = Profile; | ||||
| 		}; | ||||
| 		331C8088294A63A400263BE5 /* Debug */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			baseConfigurationReference = DC475F5210027479529644C3 /* Pods-RunnerTests.debug.xcconfig */; | ||||
| 			buildSettings = { | ||||
| 				BUNDLE_LOADER = "$(TEST_HOST)"; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				CURRENT_PROJECT_VERSION = 1; | ||||
| 				CURRENT_PROJECT_VERSION = 3; | ||||
| 				DEVELOPMENT_TEAM = L32Y3D8V83; | ||||
| 				GENERATE_INFOPLIST_FILE = YES; | ||||
| 				MARKETING_VERSION = 1.0; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | ||||
| @@ -395,10 +540,12 @@ | ||||
| 		}; | ||||
| 		331C8089294A63A400263BE5 /* Release */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			baseConfigurationReference = A565AAB9FE158487ABF3A5BF /* Pods-RunnerTests.release.xcconfig */; | ||||
| 			buildSettings = { | ||||
| 				BUNDLE_LOADER = "$(TEST_HOST)"; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				CURRENT_PROJECT_VERSION = 1; | ||||
| 				CURRENT_PROJECT_VERSION = 3; | ||||
| 				DEVELOPMENT_TEAM = L32Y3D8V83; | ||||
| 				GENERATE_INFOPLIST_FILE = YES; | ||||
| 				MARKETING_VERSION = 1.0; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | ||||
| @@ -410,10 +557,12 @@ | ||||
| 		}; | ||||
| 		331C808A294A63A400263BE5 /* Profile */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			baseConfigurationReference = 282EA28E78AB3F765E4BA719 /* Pods-RunnerTests.profile.xcconfig */; | ||||
| 			buildSettings = { | ||||
| 				BUNDLE_LOADER = "$(TEST_HOST)"; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				CURRENT_PROJECT_VERSION = 1; | ||||
| 				CURRENT_PROJECT_VERSION = 3; | ||||
| 				DEVELOPMENT_TEAM = L32Y3D8V83; | ||||
| 				GENERATE_INFOPLIST_FILE = YES; | ||||
| 				MARKETING_VERSION = 1.0; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | ||||
| @@ -447,6 +596,7 @@ | ||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
| @@ -504,6 +654,7 @@ | ||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||
| @@ -540,18 +691,34 @@ | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
| 				CLANG_ENABLE_MODULES = YES; | ||||
| 				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; | ||||
| 				CODE_SIGN_IDENTITY = "Apple Development"; | ||||
| 				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; | ||||
| 				CODE_SIGN_STYLE = Manual; | ||||
| 				CURRENT_PROJECT_VERSION = 3; | ||||
| 				DEVELOPMENT_TEAM = ""; | ||||
| 				"DEVELOPMENT_TEAM[sdk=iphoneos*]" = L32Y3D8V83; | ||||
| 				ENABLE_BITCODE = NO; | ||||
| 				INFOPLIST_FILE = Runner/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = Any.Way; | ||||
| 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel"; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.6; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 				); | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation; | ||||
| 				MARKETING_VERSION = 1.0.0; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = info.anydev.anyway; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				PROVISIONING_PROFILE_SPECIFIER = "match AppStore info.anydev.anyway"; | ||||
| 				"PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore info.anydev.anyway"; | ||||
| 				SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; | ||||
| 				SUPPORTS_MACCATALYST = NO; | ||||
| 				SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||
| 				SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||
| 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | ||||
| 				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 				VERSIONING_SYSTEM = "apple-generic"; | ||||
| 			}; | ||||
| 			name = Debug; | ||||
| @@ -562,17 +729,33 @@ | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||
| 				CLANG_ENABLE_MODULES = YES; | ||||
| 				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; | ||||
| 				CODE_SIGN_IDENTITY = "Apple Development"; | ||||
| 				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; | ||||
| 				CODE_SIGN_STYLE = Manual; | ||||
| 				CURRENT_PROJECT_VERSION = 3; | ||||
| 				DEVELOPMENT_TEAM = ""; | ||||
| 				"DEVELOPMENT_TEAM[sdk=iphoneos*]" = L32Y3D8V83; | ||||
| 				ENABLE_BITCODE = NO; | ||||
| 				INFOPLIST_FILE = Runner/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = Any.Way; | ||||
| 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel"; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.6; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 				); | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation; | ||||
| 				MARKETING_VERSION = 1.0.0; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = info.anydev.anyway; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				PROVISIONING_PROFILE_SPECIFIER = "match AppStore info.anydev.anyway"; | ||||
| 				"PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore info.anydev.anyway"; | ||||
| 				SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; | ||||
| 				SUPPORTS_MACCATALYST = NO; | ||||
| 				SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||
| 				SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||
| 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 				VERSIONING_SYSTEM = "apple-generic"; | ||||
| 			}; | ||||
| 			name = Release; | ||||
|   | ||||
| @@ -4,4 +4,7 @@ | ||||
|    <FileRef | ||||
|       location = "group:Runner.xcodeproj"> | ||||
|    </FileRef> | ||||
|    <FileRef | ||||
|       location = "group:Pods/Pods.xcodeproj"> | ||||
|    </FileRef> | ||||
| </Workspace> | ||||
|   | ||||
| @@ -1,12 +1,14 @@ | ||||
| import UIKit | ||||
| import Flutter | ||||
| import GoogleMaps | ||||
|  | ||||
| @UIApplicationMain | ||||
| @main | ||||
| @objc class AppDelegate: FlutterAppDelegate { | ||||
|   override func application( | ||||
|     _ application: UIApplication, | ||||
|     didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? | ||||
|   ) -> Bool { | ||||
|     GMSServices.provideAPIKey("IOS_GOOGLE_MAPS_API_KEY") | ||||
|     GeneratedPluginRegistrant.register(with: self) | ||||
|     return super.application(application, didFinishLaunchingWithOptions: launchOptions) | ||||
|   } | ||||
|   | ||||
| @@ -2,10 +2,12 @@ | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>CADisableMinimumFrameDurationOnPhone</key> | ||||
| 	<true/> | ||||
| 	<key>CFBundleDevelopmentRegion</key> | ||||
| 	<string>$(DEVELOPMENT_LANGUAGE)</string> | ||||
| 	<key>CFBundleDisplayName</key> | ||||
| 	<string>Fast Network Navigation</string> | ||||
| 	<string>anyway</string> | ||||
| 	<key>CFBundleExecutable</key> | ||||
| 	<string>$(EXECUTABLE_NAME)</string> | ||||
| 	<key>CFBundleIdentifier</key> | ||||
| @@ -24,6 +26,8 @@ | ||||
| 	<string>$(FLUTTER_BUILD_NUMBER)</string> | ||||
| 	<key>LSRequiresIPhoneOS</key> | ||||
| 	<true/> | ||||
| 	<key>UIApplicationSupportsIndirectInputEvents</key> | ||||
| 	<true/> | ||||
| 	<key>UILaunchStoryboardName</key> | ||||
| 	<string>LaunchScreen</string> | ||||
| 	<key>UIMainStoryboardFile</key> | ||||
| @@ -41,9 +45,38 @@ | ||||
| 		<string>UIInterfaceOrientationLandscapeLeft</string> | ||||
| 		<string>UIInterfaceOrientationLandscapeRight</string> | ||||
| 	</array> | ||||
| 	<key>CADisableMinimumFrameDurationOnPhone</key> | ||||
| 	<true/> | ||||
| 	<key>UIApplicationSupportsIndirectInputEvents</key> | ||||
| 	<true/> | ||||
| 	<key>NSLocationAlwaysAndWhenInUseUsageDescription</key> | ||||
| 	<string>$(PRODUCT_NAME) optionally uses your location to plan trips directly from your current location.</string> | ||||
| 	<key>NSLocationWhenInUseUsageDescription</key> | ||||
| 	<string>$(PRODUCT_NAME) optionally uses your location to plan trips directly from your current location.</string> | ||||
| 	<key>LSApplicationQueriesSchemes</key> | ||||
| 	<array> | ||||
| 		<!-- set by maps launcher --> | ||||
| 		<string>comgooglemaps</string> | ||||
| 		<string>baidumap</string> | ||||
| 		<string>iosamap</string> | ||||
| 		<string>waze</string> | ||||
| 		<string>yandexmaps</string> | ||||
| 		<string>yandexnavi</string> | ||||
| 		<string>citymapper</string> | ||||
| 		<string>mapswithme</string> | ||||
| 		<string>osmandmaps</string> | ||||
| 		<string>dgis</string> | ||||
| 		<string>qqmap</string> | ||||
| 		<string>here-location</string> | ||||
| 		<string>tomtomgo</string> | ||||
| 		<string>copilot</string> | ||||
| 		<string>com.sygic.aura</string> | ||||
| 		<string>nmap</string> | ||||
| 		<string>kakaomap</string> | ||||
| 		<string>tmap</string> | ||||
| 		<string>szn-mapy</string> | ||||
| 		<string>mappls</string> | ||||
| 		<!-- used by url launcher to open web browser --> | ||||
| 		<string>http</string> | ||||
| 		<string>https</string> | ||||
| 	</array> | ||||
| 	<key>ITSAppUsesNonExemptEncryption</key> | ||||
| 	<false/> | ||||
| </dict> | ||||
| </plist> | ||||
|   | ||||
							
								
								
									
										13
									
								
								frontend/ios/fastlane/.env.sample
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								frontend/ios/fastlane/.env.sample
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| # SAMPLE env file that replicates the env in the CI/CD pipeline | ||||
| # DO NOT EDIT THIS FILE | ||||
| # Copy this file to local.env and edit the values to match your local environment | ||||
| BUILD_NAME="sample" | ||||
| BUILD_NUMBER="sample" | ||||
|  | ||||
| IOS_ASC_KEY_ID="sample" | ||||
| IOS_ASC_KEY="sample" | ||||
| IOS_ASC_ISSUER_ID="sample" | ||||
| SIGNING_KEY_FILE_PATH="sample" | ||||
| SIGNING_KEY_PASSWORD="sample" | ||||
|  | ||||
| IOS_GOOGLE_MAPS_API_KEY="sample" | ||||
							
								
								
									
										8
									
								
								frontend/ios/fastlane/Appfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								frontend/ios/fastlane/Appfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| app_identifier("info.anydev.testing") # The bundle identifier of your app | ||||
| apple_id("me@moll.re") # Your Apple Developer Portal username | ||||
|  | ||||
| itc_team_id("127439860") # App Store Connect Team ID | ||||
| team_id("L32Y3D8V83") # Developer Portal Team ID | ||||
|  | ||||
| # For more information about the Appfile, see: | ||||
| #     https://docs.fastlane.tools/advanced/#appfile | ||||
							
								
								
									
										102
									
								
								frontend/ios/fastlane/Fastfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								frontend/ios/fastlane/Fastfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | ||||
| default_platform(:ios) | ||||
|  | ||||
| platform :ios do | ||||
|  | ||||
|   desc "Load the App Store Connect API token" | ||||
|   lane :load_asc_api_token do | ||||
|     app_store_connect_api_key( | ||||
|       key_id: ENV["IOS_ASC_KEY_ID"], | ||||
|       issuer_id: ENV["IOS_ASC_ISSUER_ID"], | ||||
|       key_content: ENV["IOS_ASC_KEY"], | ||||
|       is_key_content_base64: true, | ||||
|       in_house: false | ||||
|     ) | ||||
|   end | ||||
|  | ||||
|  | ||||
|   desc "Deploy a new version to closed testing (testflight)" | ||||
|   lane :deploy_testing do | ||||
|     build_name = ENV["BUILD_NAME"] | ||||
|     build_number = ENV["BUILD_NUMBER"] | ||||
|      | ||||
|     load_asc_api_token | ||||
|     api_key = lane_context[SharedValues::APP_STORE_CONNECT_API_KEY] | ||||
|  | ||||
|     sync_code_signing( | ||||
|       api_key: api_key, | ||||
|       type: "appstore", | ||||
|       readonly: true, | ||||
|     ) | ||||
|  | ||||
|  | ||||
|     sh( | ||||
|       "flutter", | ||||
|       "build", | ||||
|       "ipa", | ||||
|       "--debug", | ||||
|       "--build-name=#{build_name}", | ||||
|       "--build-number=#{build_number}", | ||||
|     ) | ||||
|  | ||||
|     # sign the app (whithout rebuilding it) | ||||
|     build_app( | ||||
|       skip_build_archive: true, | ||||
|       archive_path: "../build/ios/archive/Runner.xcarchive" | ||||
|     ) | ||||
|  | ||||
|     upload_to_testflight( | ||||
|       skip_waiting_for_build_processing: true, | ||||
|     ) | ||||
|   end | ||||
|  | ||||
|  | ||||
|   desc "Deploy a new version as a full release" | ||||
|   lane :deploy_release do | ||||
|     build_name = ENV["BUILD_NAME"] | ||||
|     build_number = ENV["BUILD_NUMBER"] | ||||
|      | ||||
|     load_asc_api_token | ||||
|     api_key = lane_context[SharedValues::APP_STORE_CONNECT_API_KEY] | ||||
|  | ||||
|     sync_code_signing( | ||||
|       api_key: api_key, | ||||
|       type: "appstore", | ||||
|       readonly: true, | ||||
|     ) | ||||
|  | ||||
|     # replace secrets by real values, the stupid way | ||||
|     sh( | ||||
|       "sed", | ||||
|       "-i", | ||||
|       "", | ||||
|       "s/IOS_GOOGLE_MAPS_API_KEY/#{ENV["IOS_GOOGLE_MAPS_API_KEY"]}/g", | ||||
|       "../Runner/AppDelegate.swift" | ||||
|  | ||||
|     ) | ||||
|  | ||||
|     sh( | ||||
|       "flutter", | ||||
|       "build", | ||||
|       "ipa", | ||||
|       "--release", | ||||
|       "--build-name=#{build_name}", | ||||
|       "--build-number=#{build_number}", | ||||
|     ) | ||||
|  | ||||
|     # sign the app (whithout rebuilding it) | ||||
|     build_app( | ||||
|       skip_build_archive: true, | ||||
|       archive_path: "../build/ios/archive/Runner.xcarchive" | ||||
|     ) | ||||
|    | ||||
|     upload_to_app_store( | ||||
|       skip_screenshots: true, | ||||
|       skip_metadata: true, | ||||
|       precheck_include_in_app_purchases: false, | ||||
|        | ||||
|       submit_for_review: true, | ||||
|       automatic_release: true, | ||||
|       # automatically release the app after review | ||||
|     ) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										8
									
								
								frontend/ios/fastlane/Matchfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								frontend/ios/fastlane/Matchfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| git_url("ssh://git@git.kluster.moll.re:2222/anydev/anyway-app-secrets.git") | ||||
|  | ||||
| storage_mode("git") | ||||
|  | ||||
| type("appstore") # The default type, can be: appstore, adhoc, enterprise or development | ||||
|  | ||||
| app_identifier(["info.anydev.anyway"]) | ||||
| username("me@moll.re") # Your Apple Developer Portal username | ||||
							
								
								
									
										48
									
								
								frontend/ios/fastlane/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								frontend/ios/fastlane/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| fastlane documentation | ||||
| ---- | ||||
|  | ||||
| # Installation | ||||
|  | ||||
| Make sure you have the latest version of the Xcode command line tools installed: | ||||
|  | ||||
| ```sh | ||||
| xcode-select --install | ||||
| ``` | ||||
|  | ||||
| For _fastlane_ installation instructions, see [Installing _fastlane_](https://docs.fastlane.tools/#installing-fastlane) | ||||
|  | ||||
| # Available Actions | ||||
|  | ||||
| ## iOS | ||||
|  | ||||
| ### ios load_asc_api_token | ||||
|  | ||||
| ```sh | ||||
| [bundle exec] fastlane ios load_asc_api_token | ||||
| ``` | ||||
|  | ||||
| Load the App Store Connect API token | ||||
|  | ||||
| ### ios deploy_testing | ||||
|  | ||||
| ```sh | ||||
| [bundle exec] fastlane ios deploy_testing | ||||
| ``` | ||||
|  | ||||
| Deploy a new version to closed testing (testflight) | ||||
|  | ||||
| ### ios deploy_release | ||||
|  | ||||
| ```sh | ||||
| [bundle exec] fastlane ios deploy_release | ||||
| ``` | ||||
|  | ||||
| Deploy a new version as a full release | ||||
|  | ||||
| ---- | ||||
|  | ||||
| This README.md is auto-generated and will be re-generated every time [_fastlane_](https://fastlane.tools) is run. | ||||
|  | ||||
| More information about _fastlane_ can be found on [fastlane.tools](https://fastlane.tools). | ||||
|  | ||||
| The documentation of _fastlane_ can be found on [docs.fastlane.tools](https://docs.fastlane.tools). | ||||
| @@ -1 +1,2 @@ | ||||
| #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig" | ||||
| #include "ephemeral/Flutter-Generated.xcconfig" | ||||
|   | ||||
| @@ -1 +1,2 @@ | ||||
| #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig" | ||||
| #include "ephemeral/Flutter-Generated.xcconfig" | ||||
|   | ||||
							
								
								
									
										43
									
								
								frontend/macos/Podfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								frontend/macos/Podfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| platform :osx, '10.14' | ||||
|  | ||||
| # CocoaPods analytics sends network stats synchronously affecting flutter build latency. | ||||
| ENV['COCOAPODS_DISABLE_STATS'] = 'true' | ||||
|  | ||||
| project 'Runner', { | ||||
|   'Debug' => :debug, | ||||
|   'Profile' => :release, | ||||
|   'Release' => :release, | ||||
| } | ||||
|  | ||||
| def flutter_root | ||||
|   generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'ephemeral', 'Flutter-Generated.xcconfig'), __FILE__) | ||||
|   unless File.exist?(generated_xcode_build_settings_path) | ||||
|     raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure \"flutter pub get\" is executed first" | ||||
|   end | ||||
|  | ||||
|   File.foreach(generated_xcode_build_settings_path) do |line| | ||||
|     matches = line.match(/FLUTTER_ROOT\=(.*)/) | ||||
|     return matches[1].strip if matches | ||||
|   end | ||||
|   raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Flutter-Generated.xcconfig, then run \"flutter pub get\"" | ||||
| end | ||||
|  | ||||
| require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root) | ||||
|  | ||||
| flutter_macos_podfile_setup | ||||
|  | ||||
| target 'Runner' do | ||||
|   use_frameworks! | ||||
|   use_modular_headers! | ||||
|  | ||||
|   flutter_install_all_macos_pods File.dirname(File.realpath(__FILE__)) | ||||
|   target 'RunnerTests' do | ||||
|     inherit! :search_paths | ||||
|   end | ||||
| end | ||||
|  | ||||
| post_install do |installer| | ||||
|   installer.pods_project.targets.each do |target| | ||||
|     flutter_additional_macos_build_settings(target) | ||||
|   end | ||||
| end | ||||
		Reference in New Issue
	
	Block a user