Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[iOS] Third-party libraries not found for "staging" config: Solution! #426

Closed
danieldunderfelt opened this issue Jul 12, 2016 · 39 comments
Closed

Comments

@danieldunderfelt
Copy link

Hi!

I installed CodePush in my app today and followed the instructions for multi-environments, where the docs say to create a "staging" project configuration. After doing that, the project would not build at all when the scheme was set to "staging", complaining that no libraries could be found. Re-linking with rnpm didn't help, but I figured out that the issue was that no third-party libraries (including code-push!) include the staging configuration, making the build fail.

The solution was this stackoverflow answer: http://stackoverflow.com/questions/30884778/how-to-compile-a-project-with-app-and-library-in-the-same-workspace-with-differe/30884779#30884779

I had to change it a bit, using the same string for "header search paths" as "framework search paths" (functionally omitting the /include part of that string), but now everything works.

I think this should be mentioned in the documentation right where the user is instructed to create the "staging" configuration.

@geof90
Copy link
Contributor

geof90 commented Jul 12, 2016

@danieldunderfelt Thanks for the feedback. I will see how we can improve the documentation.

When you created the staging project configuration, did you also manually create a Staging scheme?

@danieldunderfelt
Copy link
Author

Hi!

No, that was created when I copied the release config and made the staging config.

@geof90
Copy link
Contributor

geof90 commented Jul 13, 2016

@danieldunderfelt Unfortunately I was unable to repro the error you were seeing.
After duplicating the release config and calling it staging + setting up the CODEPUSH_KEY config, all I did was go to Product > Scheme > Edit Scheme then set the Build Configuration to Staging. After that I ran the app and it was sync-ing against CodePush with the Staging key just fine.

What version of XCode are you on? I don't know if that matters but I'm using v7.1.1.
Also are you using CocoaPods for dependency management?

@danieldunderfelt
Copy link
Author

@geof90 No, not using CocoaPods. Does it manage these things for me?

When I figured out why my builds weren't working, that the third-party packages didn't have a staging config, I thought having cocoapods wouldn't matter.

@geof90
Copy link
Contributor

geof90 commented Jul 14, 2016

@danieldunderfelt no I only asked because I wasn't using CocoaPods and I was wondering if you were.

@danieldunderfelt
Copy link
Author

@geof90 oh, good good!

Maybe your xcode made staging config for your third-party packages, and my xcode glitched out and didn't?

@thorbenandresen
Copy link

thorbenandresen commented Jul 15, 2016

Nice, this did also fix it for me. I omitted the Step 3 (changing the project.pbxproj) and my Fabric.framework does still work fine.

However when I use different PRODUCT_NAMES for the different build environment, I am getting now a build error. I am have setup the names according to this article

screen shot 2016-07-15 at 1 20 49 pm

When i build my release config everything works, but not when I build the debug or staging config. In these cases I get the following errors:

screen shot 2016-07-15 at 11 22 42 am

I have tried to fix it by

  • set Target => Info => Executable file to $(PRODUCT_NAME)
  • set Target => Bundler Loader to => $(BUILT_PRODUCTS_DIR)/$(PRODUCT_NAME).app/$(PRODUCT_NAME)

I have cleaned my build folder before each build.

This did not happen before, so it could be related to this issue.

(For now my workaround is to use different app icons in order to distinguish my different build environment - as outlines in the article above)

@FernandoMorais
Copy link

Here I solved the staging problem by correcting the TEST_HOST which is pointing to a fixed value despite the new PRODUCT_NAME.

I've changed the scheme to use the staging variation for tests builds.

I hope it helps!

@huitsing
Copy link

Our solution might be helpful here as well: https://zeemee.engineering/how-to-set-up-multiple-schemes-configurations-in-xcode-for-your-react-native-ios-app-7da4b5237966#.pxa41apg1

@FernandoMorais
Copy link

really nice @huitsing! Much more complete solution!

@jm555jm
Copy link

jm555jm commented Aug 18, 2016

I have the same issue, too. I use xcode 8 beta 6.
library not found for -lCodePush with build configuration "Staging".

I have fix just re-add libCodePush.a at Linked Framework and Libraries.
Now everything works fine =).

@davidruisinger
Copy link

@jm555jm THX for the hint. I had quite a few external libraries and I was desperately trying to unlink and re-link them without any luck. But manually removing and re-adding them in the Link Binary with Libraries section finally solved the issue and my builds are succeeding again.

@bernatfortet
Copy link

@jm555jm @flavordaaave thanks for the tip!
It took me a couple minutes to figure out exactly how to do that, so for others:

  1. Select your app target
  2. Make sure you are in the General Tab
  3. Scroll down until Link Binary with Libraries
  4. Select libCodePush
  5. Press - minus button at the bottom left
  6. Press + plus button and search for libCodePush
    👍

@bernatfortet
Copy link

Actually after trying that I found that I was still with the Release scheme.
After changing it to staging, and after removing and re-adding the library it still doesn't work 😞
Same exact error.

@Amurmurmur
Copy link

+1 Same error, with the header, when on Staging Scheme :(

@sergey-akhalkov
Copy link
Contributor

sergey-akhalkov commented Feb 8, 2017

@bernatfortet, could you please try to apply instructions from this PR #690 and let me know if it helps?

@sergey-akhalkov
Copy link
Contributor

Closing this due to #690 has been merged and fixed similar issue #688, please feel free to reopen in case of any questions or issues.

@ndbroadbent
Copy link

Hi, I also ran into an issue, and came straight here. But for me, I just had to run pod install, and then everything worked after that.

@ndbroadbent
Copy link

ndbroadbent commented Apr 19, 2017

Argh I still struggled with this for quite a while, but then I realized that you need to build a Release version first, before building Staging. (Because I changed the 'build products path' to $(BUILD_DIR)/Release$(EFFECTIVE_PLATFORM_NAME).)

So if you run a clean, or delete derived data, make sure you switch to your Release scheme and press cmd+B to run a build, before you try to build Staging. Because your Staging build depends on some things that need to be in that Release folder.

@fannt
Copy link

fannt commented May 24, 2017

I had a trouble with cocoapods overriding this config settings, used this as a workaround:

post_install do |installer|
  installer.pods_project.targets.each do |target|
    if target.name == 'YourTarget'
      target.build_configurations.each do |config|
        if config.name == 'STAGING'
          config.build_settings['CONFIGURATION_TEMP_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
          config.build_settings['CONFIGURATION_BUILD_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
        end
      end
    end
  end
end

@mtt87
Copy link

mtt87 commented Sep 16, 2017

In my case after following the instructions on the official documentation I had to change this as well to be $PODS_BUILD_DIR/Release$(EFFECTIVE_PLATFORM_NAME)
screen shot 2017-09-16 at 17 38 37

@wprater
Copy link

wprater commented Nov 19, 2017

These fixes don't seem to work on CI servers, since the Release build path won't have any build or header files in it.

Any thoughts on fixes or running on a CI (Buddy Build or MSFT App Center?)

@antsmo
Copy link

antsmo commented Feb 6, 2018

@wprater I'm having the same issue, it works fine if I build Release first. Did you happen to get it working?

@aligfl
Copy link

aligfl commented May 10, 2018

@wprater did you get it working on your CI system ? aka buddybuild or App Center?
Did you try the solution proposed by @huitsing ?

aka: https://zeemee.engineering/how-to-set-up-multiple-schemes-configurations-in-xcode-for-your-react-native-ios-app-7da4b5237966

@jpaolini
Copy link

Not sure if this helps, but I had to do something similar to @fannt. Cocopods was overwriting the configurations:

post_install do |installer|
  installer.pods_project.targets.each do |target|
    if target.name.downcase == 'YourTargetName
      target.build_configurations.each do |config|
        if config.name.downcase == 'staging'
          config.build_settings['CONFIGURATION_BUILD_DIR'] = '$(BUILD_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
        end
      end
    end
  end
end

@aligfl
Copy link

aligfl commented May 11, 2018

Someone posted a solution on another thread. Which has worked for me locally + CI (buddybuild)

Solution: facebook/react-native#11813 (comment)

@Amurmurmur
Copy link

Amurmurmur commented Jun 23, 2018

Any one having an issue when using tipsi-stripe and having multiple deployments?
My pod file

# Uncomment the next line to define a global platform for your project
platform :ios, '9.0'

workspace 'appname'

target 'targetNameTests' do
  # Uncomment the next line if you're using Swift or would like to use dynamic frameworks
  # use_frameworks!

  # Pods for targetName
  pod 'AppCenter/Crashes', '~> 1.6.1'
  pod 'AppCenter/Analytics', '~> 1.6.1'
  pod 'AppCenterReactNativeShared', '~> 1.5.1'
  # Stripe
  pod 'Stripe', '~> 11.2.0'

    target 'targetNameTests' do
        inherit! :search_paths
        # Pods for testing
    end
end

post_install do |installer|
    installer.pods_project.targets.each do |target|
      if target.name == 'targetName'
        target.build_configurations.each do |config|
          if config.name == 'Staging'
            config.build_settings['CONFIGURATION_TEMP_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
            config.build_settings['CONFIGURATION_BUILD_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
          end
        end
      end
    end
  end

I've set Per-configuration Build Products Path for Staging to $(BUILD_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)
Also I've updated PODS_CONFIGURATION_BUILD_DIR for Staging to ${PODS_BUILD_DIR}/Release$(EFFECTIVE_PLATFORM_NAME)

Im getting the following:

ld: library not found for -lStripe
clang: error: linker command failed with exit code 1 (use -v to see invocation)

@NickToropov
Copy link
Contributor

Hi @Amurmurmur, I'm sure that problem is still in CONFIGURATION_BUILD_DIR and CONFIGURATION_TEMP_DIR settings. I've just created new ReactNative application, copied your Podfile, actualized it to this:

# Uncomment the next line to define a global platform for your project
platform :ios, '9.0'

workspace 'appname'

target 'Issue426' do
  # Uncomment the next line if you're using Swift or would like to use dynamic frameworks
  # use_frameworks!

  # React Native requirements
pod 'React', :path => '../node_modules/react-native', :subspecs => [
    'Core',
    'CxxBridge', # Include this for RN >= 0.47
    'DevSupport', # Include this to enable In-App Devmenu if RN >= 0.43
    'RCTText',
    'RCTNetwork',
    'RCTWebSocket', # Needed for debugging
    'RCTAnimation', # Needed for FlatList and animations running on native UI thread
    # Add any other subspecs you want to use in your project
 ]
 # Explicitly include Yoga if you are using RN >= 0.42.0
 pod 'yoga', :path => '../node_modules/react-native/ReactCommon/yoga'
 pod 'DoubleConversion', :podspec => '../node_modules/react-native/third-party-podspecs/DoubleConversion.podspec'
 pod 'glog', :podspec => '../node_modules/react-native/third-party-podspecs/glog.podspec'
 pod 'Folly', :podspec => '../node_modules/react-native/third-party-podspecs/Folly.podspec'

  # Pods for targetName
  pod 'AppCenter/Crashes', '~> 1.6.1'
  pod 'AppCenter/Analytics', '~> 1.6.1'
  pod 'AppCenterReactNativeShared', '~> 1.5.1'
  # Stripe
  pod 'Stripe', '~> 11.2.0'

end

post_install do |installer|
    installer.pods_project.targets.each do |target|
        if target.name == 'Issue426'
            target.build_configurations.each do |config|
                if config.name == 'Staging'
                    config.build_settings['CONFIGURATION_TEMP_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
                    config.build_settings['CONFIGURATION_BUILD_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
                end
            end
        end
    end
end

then executed pod install, opened generated appname.xcworkspace in XCode, added new Staging configuration (based on Release configuration), added new Staging scheme, updated Pre-configuration Build Files Path and Pre-configuration Intermediate Build Files Path for both Issue426 and Pods project. After this steps my demo project built successfully.

Please try my steps and if the issue won't be solved provide me with sample project that I could able to reproduce your situation and help to fix it.

@jerson
Copy link

jerson commented Jul 7, 2018

for me worked with

post_install do |installer|
  installer.pods_project.build_configurations.each do |config|
              if config.name == 'Staging'
                  config.build_settings['CONFIGURATION_TEMP_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
                  config.build_settings['CONFIGURATION_BUILD_DIR'] = '$(BUILD_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
              end
  end
  installer.pods_project.targets.each do |target|
          target.build_configurations.each do |config|
              if config.name == 'Staging'
                  config.build_settings['CONFIGURATION_TEMP_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
                  config.build_settings['PODS_CONFIGURATION_BUILD_DIR'] = '${PODS_BUILD_DIR}/Release$(EFFECTIVE_PLATFORM_NAME)'
              end
          end
  end
end

@pppluto
Copy link

pppluto commented Nov 10, 2018

for me i just need set like this

post_install do |installer|
    installer.pods_project.build_configurations.each do |config|
        if config.name == 'Staging'
            config.build_settings['CONFIGURATION_BUILD_DIR'] = '$(BUILD_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
        end
    end
end

@AlgirdasVZ
Copy link

If anybody is still struggling with this issue - there is a simple way to differentiate between Codepush staging and release keys using the same "Release" configuration if you already do your builds with Fastlane.

You basically need to have these lines in your Info.plist:

<key>CodePushDeploymentKey</key>  
<string>$(CODEPUSH_KEY)</string>

and then, assuming you already have different Fastlane lanes for staging and release, add appropriate Codepush key for each lane in form of a gym param:

xcargs: "CODEPUSH_KEY=<your staging/release key>"

The key will be inserted during build process and that's about it!
This approach definitely does not cover all cases, but could be useful for those who are searching for a simplified solution without the need to edit your xcode project file.

@nikopolidi
Copy link

nikopolidi commented Aug 30, 2019

I got it solved and implemented for several projects.
Here is what I did:

  1. add that snippet to the end of Podfile:
post_install do |installer|
  installer.pods_project.build_configurations.each do |config|
              if config.name == 'Staging'
                  config.build_settings['CONFIGURATION_TEMP_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
                  config.build_settings['CONFIGURATION_BUILD_DIR'] = '$(BUILD_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
              end
  end
  installer.pods_project.targets.each do |target|
          target.build_configurations.each do |config|
              if config.name == 'Staging'
                  config.build_settings['CONFIGURATION_TEMP_DIR'] = '$(PROJECT_TEMP_DIR)/Release$(EFFECTIVE_PLATFORM_NAME)'
                  config.build_settings['PODS_CONFIGURATION_BUILD_DIR'] = '${PODS_BUILD_DIR}/Release$(EFFECTIVE_PLATFORM_NAME)'
              end
          end
  end
end
  1. Update build paths:

Set Staging value for Per-configuration Build Products Path and Per-configuration intermediate Build Files path to $(BUILD_DIR)/Release$(EFFECTIVE_PLATFORM_NAME) so it looks the same as Release one

  1. Reinstall pods.

From project root: cd ios && pod deintegrate && pod install

  1. done. Try to archive with that target to test

NOTE: In AppCenter I was having build failures when it's archiving successfully locally for staging target. I got it solved by removing build config from that branch and adding it back, seems like it has some kind of caching issues idk

@JimmyDLA
Copy link

JimmyDLA commented Dec 10, 2019

can anyone tell me where I can find my BUILD_DIR and EFFECTIVE_PLATFORM_NAME?
@nikopolidi @MrPluto @jerson @sergey-akhalkov @Almouro @Amurmurmur,

@Amurmurmur
Copy link

@JimmyDLA You can find it in Xcode
https://stackoverflow.com/a/11657874

@chj-damon
Copy link

none of these answers solve my problem.

@TwR-Slice
Copy link

Just an FYI what worked in my case:

After creating the new configuration Staging, simply run pod deintegrate && pod install inside the ios folder of your Project. CocoaPods will then create all configurations on its own.

Staging Build -> Success
Staging Archive -> Success

Release Build -> Success
Release Archive -> Success

@tabrindle
Copy link

tabrindle commented Apr 16, 2021

For what it's worth, there are a few easier ways to handle this issue, after attempting without success to add a configuration, I decided it was overkill for a single value change.

Method 1: Use info.plist variable substitution, xcargs a and a default value (post xcode 11.4)

  • Set the deployment key to $(CODEPUSH_KEY:default=209348572039487520934875) in info.plist
  • We use fastlane, so calling gym/build_app with xcargs injected in from env looks like this:

build_app(scheme: "Release", export_method: "app-store", xcargs: "CODEPUSH_KEY='#{ENV["CODEPUSH_KEY"]}'"

Method 2: Use a Run Script

  • Target -> Build Phases -> Click plus -> New Run Script Phase.
  • add something like this: plutil -replace CodePushDeploymentKey -string $CODEPUSH_KEY ./APPNAME/Info.plist

Method 3: Set in JavaScript

  • did you know there is a setDeploymentKey method in the plugin for both iOS and Android? No? Cool, wasn't just me.
  • Turns out you can just pass in some configuration when starting your app, like this. This way the keys can live in your code, or however you are dealing with env vars on the javascript side. We use react-native-config, and have a staging and production env file.
const Application = __DEV__
  ? App
  : codePush({
      deploymentKey:
        Platform.OS === 'ios'
          ? config.IOS_CODE_PUSH_DEPLOYMENT_KEY
          : config.ANDROID_CODE_PUSH_DEPLOYMENT_KEY,
    })(App);

export default Application;

Voila, it works. And you can go back to hating Apple, Xcode and everything associated with them in peace.

@MRezaBaiat
Copy link

i had the same problem but went away after i ran pod install --repo-update

@Gamote
Copy link

Gamote commented May 31, 2023

Just an FYI what worked in my case:

After creating the new configuration Staging, simply run pod deintegrate && pod install inside the ios folder of your Project. CocoaPods will then create all configurations on its own.

Staging Build -> Success Staging Archive -> Success

Release Build -> Success Release Archive -> Success

After following the Multi-Deployment Testing guide we encounter similar errors. They were fixed by pod deintegrate && pod install. Thank you.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests