keycloak-aplcache
Changes
dependencies/server-all/pom.xml 5(+5 -0)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-facebook-ext.html 0(+0 -0)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-github-ext.html 0(+0 -0)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-google-ext.html 0(+0 -0)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-linkedin-ext.html 0(+0 -0)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-social.html 3(+2 -1)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow.html 1(+1 -0)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow-ext.html 7(+7 -0)
forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-twitter-ext.html 0(+0 -0)
social/stackoverflow/.gitignore 1(+1 -0)
social/stackoverflow/pom.xml 40(+40 -0)
social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackoverflowIdentityProvider.java 313(+313 -0)
social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackOverflowIdentityProviderConfig.java 40(+40 -0)
Details
dependencies/server-all/pom.xml 5(+5 -0)
diff --git a/dependencies/server-all/pom.xml b/dependencies/server-all/pom.xml
index a7bd522..2c1f6f3 100755
--- a/dependencies/server-all/pom.xml
+++ b/dependencies/server-all/pom.xml
@@ -127,6 +127,11 @@
<artifactId>keycloak-social-linkedin</artifactId>
<version>${project.version}</version>
</dependency>
+ <dependency>
+ <groupId>org.keycloak</groupId>
+ <artifactId>keycloak-social-stackoverflow</artifactId>
+ <version>${project.version}</version>
+ </dependency>
<!-- ldap federation api -->
<dependency>
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-facebook-ext.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-facebook-ext.html
new file mode 100755
index 0000000..e69de29
--- /dev/null
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-facebook-ext.html
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-github-ext.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-github-ext.html
new file mode 100755
index 0000000..e69de29
--- /dev/null
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-github-ext.html
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-google-ext.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-google-ext.html
new file mode 100755
index 0000000..e69de29
--- /dev/null
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-google-ext.html
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-linkedin-ext.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-linkedin-ext.html
new file mode 100755
index 0000000..e69de29
--- /dev/null
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-linkedin-ext.html
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-social.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-social.html
index d37e6a3..9646174 100755
--- a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-social.html
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-social.html
@@ -28,12 +28,13 @@
</div>
<span tooltip-placement="right" tooltip="The client or application secret registered withing the identity provider." class="fa fa-info-circle"></span>
</div>
+ <div data-ng-include data-src="resourceUrl + '/partials/realm-identity-provider-' + identityProvider.providerId + '-ext.html'"></div>
<div class="form-group clearfix">
<label class="col-sm-2 control-label" for="defaultScope">Default Scopes </label>
<div class="col-sm-4">
<input class="form-control" id="defaultScope" type="text" ng-model="identityProvider.config.defaultScope">
</div>
- <span tooltip-placement="right" tooltip="The scopes to be sent when asking for authorization. It can be a space-separated list of scopes. Defaults to 'openid'." class="fa fa-info-circle"></span>
+ <span tooltip-placement="right" tooltip="The scopes to be sent when asking for authorization. See documentation for possible values, separator and default value'." class="fa fa-info-circle"></span>
</div>
<div class="form-group">
<label class="col-sm-2 control-label" for="enabled">Store Tokens</label>
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow.html
new file mode 100755
index 0000000..a4630ac
--- /dev/null
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow.html
@@ -0,0 +1 @@
+<div data-ng-include data-src="resourceUrl + '/partials/realm-identity-provider-social.html'"></div>
\ No newline at end of file
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow-ext.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow-ext.html
new file mode 100755
index 0000000..86516df
--- /dev/null
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-stackoverflow-ext.html
@@ -0,0 +1,7 @@
+ <div class="form-group clearfix">
+ <label class="col-sm-2 control-label" for="clientId">Key <span class="required">*</span></label>
+ <div class="col-sm-4">
+ <input class="form-control" id="clientId" type="text" ng-model="identityProvider.config.key" required>
+ </div>
+ <span tooltip-placement="right" tooltip="The Key obtained from Stack Overflow application registration." class="fa fa-info-circle"></span>
+ </div>
diff --git a/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-twitter-ext.html b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-twitter-ext.html
new file mode 100755
index 0000000..e69de29
--- /dev/null
+++ b/forms/common-themes/src/main/resources/theme/base/admin/resources/partials/realm-identity-provider-twitter-ext.html
social/stackoverflow/.gitignore 1(+1 -0)
diff --git a/social/stackoverflow/.gitignore b/social/stackoverflow/.gitignore
new file mode 100644
index 0000000..b83d222
--- /dev/null
+++ b/social/stackoverflow/.gitignore
@@ -0,0 +1 @@
+/target/
social/stackoverflow/pom.xml 40(+40 -0)
diff --git a/social/stackoverflow/pom.xml b/social/stackoverflow/pom.xml
new file mode 100755
index 0000000..f05a77b
--- /dev/null
+++ b/social/stackoverflow/pom.xml
@@ -0,0 +1,40 @@
+<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+ xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
+ <parent>
+ <artifactId>keycloak-social-parent</artifactId>
+ <groupId>org.keycloak</groupId>
+ <version>1.2.0.Beta1-SNAPSHOT</version>
+ <relativePath>../pom.xml</relativePath>
+ </parent>
+ <modelVersion>4.0.0</modelVersion>
+ <packaging>jar</packaging>
+
+ <artifactId>keycloak-social-stackoverflow</artifactId>
+ <name>Keycloak Social StackOverflow</name>
+ <description/>
+
+ <dependencies>
+ <dependency>
+ <groupId>org.keycloak</groupId>
+ <artifactId>keycloak-social-core</artifactId>
+ <version>${project.version}</version>
+ <scope>provided</scope>
+ </dependency>
+ <dependency>
+ <groupId>org.keycloak</groupId>
+ <artifactId>keycloak-broker-oidc</artifactId>
+ <version>${project.version}</version>
+ <scope>provided</scope>
+ </dependency>
+ <dependency>
+ <groupId>org.codehaus.jackson</groupId>
+ <artifactId>jackson-mapper-asl</artifactId>
+ <scope>provided</scope>
+ </dependency>
+ <dependency>
+ <groupId>org.jboss.logging</groupId>
+ <artifactId>jboss-logging</artifactId>
+ <scope>provided</scope>
+ </dependency>
+ </dependencies>
+</project>
diff --git a/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackoverflowIdentityProvider.java b/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackoverflowIdentityProvider.java
new file mode 100755
index 0000000..b1cf142
--- /dev/null
+++ b/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackoverflowIdentityProvider.java
@@ -0,0 +1,313 @@
+/*
+ * JBoss, Home of Professional Open Source
+ *
+ * Copyright 2015 Red Hat, Inc. and/or its affiliates.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.keycloak.social.stackoverflow;
+
+import java.io.StringWriter;
+import java.net.MalformedURLException;
+import java.net.URL;
+import java.net.URLDecoder;
+import java.util.HashMap;
+
+import org.codehaus.jackson.JsonNode;
+import org.jboss.logging.Logger;
+import org.keycloak.broker.oidc.AbstractOAuth2IdentityProvider;
+import org.keycloak.broker.oidc.util.SimpleHttp;
+import org.keycloak.broker.provider.FederatedIdentity;
+import org.keycloak.broker.provider.IdentityBrokerException;
+import org.keycloak.social.SocialIdentityProvider;
+
+/**
+ * Stackoverflow social provider. See https://developer.linkedin.com/docs/oauth2
+ *
+ * @author Vlastimil Elias (velias at redhat dot com)
+ */
+public class StackoverflowIdentityProvider extends AbstractOAuth2IdentityProvider<StackOverflowIdentityProviderConfig>
+ implements SocialIdentityProvider<StackOverflowIdentityProviderConfig> {
+
+ private static final Logger log = Logger.getLogger(StackoverflowIdentityProvider.class);
+
+ public static final String AUTH_URL = "https://stackexchange.com/oauth";
+ public static final String TOKEN_URL = "https://stackexchange.com/oauth/access_token";
+ public static final String PROFILE_URL = "https://api.stackexchange.com/2.2/me?order=desc&sort=name&site=stackoverflow";
+ public static final String DEFAULT_SCOPE = "";
+
+ public StackoverflowIdentityProvider(StackOverflowIdentityProviderConfig config) {
+ super(config);
+ config.setAuthorizationUrl(AUTH_URL);
+ config.setTokenUrl(TOKEN_URL);
+ config.setUserInfoUrl(PROFILE_URL);
+ }
+
+ @Override
+ protected FederatedIdentity doGetFederatedIdentity(String accessToken) {
+ log.debug("doGetFederatedIdentity()");
+ try {
+
+ String URL = PROFILE_URL + "&access_token=" + accessToken + "&key=" + getConfig().getKey();
+ if (log.isDebugEnabled()) {
+ log.debug("StackOverflow profile request to: " + URL);
+ }
+ JsonNode profile = SimpleHttp.doGet(URL).asJson().get("items").get(0);
+
+ FederatedIdentity user = new FederatedIdentity(getJsonProperty(profile, "user_id"));
+
+ user.setUsername(extractUsernameFromProfileURL(getJsonProperty(profile, "link")));
+ // TODO username contains html encoding of national chracters sometimes
+ user.setName(unescapeHtml3(getJsonProperty(profile, "display_name")));
+ // email is not provided
+ // user.setEmail(getJsonProperty(profile, "email"));
+
+ return user;
+ } catch (Exception e) {
+ throw new IdentityBrokerException("Could not obtain user profile from Stackoverflow.", e);
+ }
+ }
+
+ protected static String extractUsernameFromProfileURL(String profileURL) {
+ if (isNotBlank(profileURL)) {
+
+ try {
+ log.debug("go to extract username from profile URL " + profileURL);
+ URL u = new URL(profileURL);
+ String path = u.getPath();
+ if (isNotBlank(path) && path.length() > 1) {
+ if (path.startsWith("/")) {
+ path = path.substring(1);
+ }
+ String[] pe = path.split("/");
+ if (pe.length >= 3) {
+ return URLDecoder.decode(pe[2], "UTF-8");
+ } else {
+ log.warn("Stackoverflow profile URL path is without second part: " + profileURL);
+ }
+ } else {
+ log.warn("Stackoverflow profile URL is without path part: " + profileURL);
+ }
+ } catch (MalformedURLException e) {
+ log.warn("Stackoverflow profile URL is malformed: " + profileURL);
+ } catch (Exception e) {
+ log.warn("Stackoverflow profile URL " + profileURL + " username extraction failed due: " + e.getMessage());
+ }
+ }
+ return null;
+ }
+
+ private static boolean isNotBlank(String s) {
+ return s != null && s.trim().length() > 0;
+ }
+
+ @Override
+ protected String getDefaultScopes() {
+ return DEFAULT_SCOPE;
+ }
+
+ public static final String unescapeHtml3(final String input) {
+ if (input == null)
+ return null;
+ StringWriter writer = null;
+ int len = input.length();
+ int i = 1;
+ int st = 0;
+ while (true) {
+ // look for '&'
+ while (i < len && input.charAt(i - 1) != '&')
+ i++;
+ if (i >= len)
+ break;
+
+ // found '&', look for ';'
+ int j = i;
+ while (j < len && j < i + MAX_ESCAPE + 1 && input.charAt(j) != ';')
+ j++;
+ if (j == len || j < i + MIN_ESCAPE || j == i + MAX_ESCAPE + 1) {
+ i++;
+ continue;
+ }
+
+ // found escape
+ if (input.charAt(i) == '#') {
+ // numeric escape
+ int k = i + 1;
+ int radix = 10;
+
+ final char firstChar = input.charAt(k);
+ if (firstChar == 'x' || firstChar == 'X') {
+ k++;
+ radix = 16;
+ }
+
+ try {
+ int entityValue = Integer.parseInt(input.substring(k, j), radix);
+
+ if (writer == null)
+ writer = new StringWriter(input.length());
+ writer.append(input.substring(st, i - 1));
+
+ if (entityValue > 0xFFFF) {
+ final char[] chrs = Character.toChars(entityValue);
+ writer.write(chrs[0]);
+ writer.write(chrs[1]);
+ } else {
+ writer.write(entityValue);
+ }
+
+ } catch (NumberFormatException ex) {
+ i++;
+ continue;
+ }
+ } else {
+ // named escape
+ CharSequence value = lookupMap.get(input.substring(i, j));
+ if (value == null) {
+ i++;
+ continue;
+ }
+
+ if (writer == null)
+ writer = new StringWriter(input.length());
+ writer.append(input.substring(st, i - 1));
+
+ writer.append(value);
+ }
+
+ // skip escape
+ st = j + 1;
+ i = st;
+ }
+
+ if (writer != null) {
+ writer.append(input.substring(st, len));
+ return writer.toString();
+ }
+ return input;
+ }
+
+ private static final String[][] ESCAPES = { { "\"", "quot" }, // " - double-quote
+ { "&", "amp" }, // & - ampersand
+ { "<", "lt" }, // < - less-than
+ { ">", "gt" }, // > - greater-than
+
+ // Mapping to escape ISO-8859-1 characters to their named HTML 3.x equivalents.
+ { "\u00A0", "nbsp" }, // non-breaking space
+ { "\u00A1", "iexcl" }, // inverted exclamation mark
+ { "\u00A2", "cent" }, // cent sign
+ { "\u00A3", "pound" }, // pound sign
+ { "\u00A4", "curren" }, // currency sign
+ { "\u00A5", "yen" }, // yen sign = yuan sign
+ { "\u00A6", "brvbar" }, // broken bar = broken vertical bar
+ { "\u00A7", "sect" }, // section sign
+ { "\u00A8", "uml" }, // diaeresis = spacing diaeresis
+ { "\u00A9", "copy" }, // © - copyright sign
+ { "\u00AA", "ordf" }, // feminine ordinal indicator
+ { "\u00AB", "laquo" }, // left-pointing double angle quotation mark = left pointing guillemet
+ { "\u00AC", "not" }, // not sign
+ { "\u00AD", "shy" }, // soft hyphen = discretionary hyphen
+ { "\u00AE", "reg" }, // ® - registered trademark sign
+ { "\u00AF", "macr" }, // macron = spacing macron = overline = APL overbar
+ { "\u00B0", "deg" }, // degree sign
+ { "\u00B1", "plusmn" }, // plus-minus sign = plus-or-minus sign
+ { "\u00B2", "sup2" }, // superscript two = superscript digit two = squared
+ { "\u00B3", "sup3" }, // superscript three = superscript digit three = cubed
+ { "\u00B4", "acute" }, // acute accent = spacing acute
+ { "\u00B5", "micro" }, // micro sign
+ { "\u00B6", "para" }, // pilcrow sign = paragraph sign
+ { "\u00B7", "middot" }, // middle dot = Georgian comma = Greek middle dot
+ { "\u00B8", "cedil" }, // cedilla = spacing cedilla
+ { "\u00B9", "sup1" }, // superscript one = superscript digit one
+ { "\u00BA", "ordm" }, // masculine ordinal indicator
+ { "\u00BB", "raquo" }, // right-pointing double angle quotation mark = right pointing guillemet
+ { "\u00BC", "frac14" }, // vulgar fraction one quarter = fraction one quarter
+ { "\u00BD", "frac12" }, // vulgar fraction one half = fraction one half
+ { "\u00BE", "frac34" }, // vulgar fraction three quarters = fraction three quarters
+ { "\u00BF", "iquest" }, // inverted question mark = turned question mark
+ { "\u00C0", "Agrave" }, // А - uppercase A, grave accent
+ { "\u00C1", "Aacute" }, // Б - uppercase A, acute accent
+ { "\u00C2", "Acirc" }, // В - uppercase A, circumflex accent
+ { "\u00C3", "Atilde" }, // Г - uppercase A, tilde
+ { "\u00C4", "Auml" }, // Д - uppercase A, umlaut
+ { "\u00C5", "Aring" }, // Е - uppercase A, ring
+ { "\u00C6", "AElig" }, // Ж - uppercase AE
+ { "\u00C7", "Ccedil" }, // З - uppercase C, cedilla
+ { "\u00C8", "Egrave" }, // И - uppercase E, grave accent
+ { "\u00C9", "Eacute" }, // Й - uppercase E, acute accent
+ { "\u00CA", "Ecirc" }, // К - uppercase E, circumflex accent
+ { "\u00CB", "Euml" }, // Л - uppercase E, umlaut
+ { "\u00CC", "Igrave" }, // М - uppercase I, grave accent
+ { "\u00CD", "Iacute" }, // Н - uppercase I, acute accent
+ { "\u00CE", "Icirc" }, // О - uppercase I, circumflex accent
+ { "\u00CF", "Iuml" }, // П - uppercase I, umlaut
+ { "\u00D0", "ETH" }, // Р - uppercase Eth, Icelandic
+ { "\u00D1", "Ntilde" }, // С - uppercase N, tilde
+ { "\u00D2", "Ograve" }, // Т - uppercase O, grave accent
+ { "\u00D3", "Oacute" }, // У - uppercase O, acute accent
+ { "\u00D4", "Ocirc" }, // Ф - uppercase O, circumflex accent
+ { "\u00D5", "Otilde" }, // Х - uppercase O, tilde
+ { "\u00D6", "Ouml" }, // Ц - uppercase O, umlaut
+ { "\u00D7", "times" }, // multiplication sign
+ { "\u00D8", "Oslash" }, // Ш - uppercase O, slash
+ { "\u00D9", "Ugrave" }, // Щ - uppercase U, grave accent
+ { "\u00DA", "Uacute" }, // Ъ - uppercase U, acute accent
+ { "\u00DB", "Ucirc" }, // Ы - uppercase U, circumflex accent
+ { "\u00DC", "Uuml" }, // Ь - uppercase U, umlaut
+ { "\u00DD", "Yacute" }, // Э - uppercase Y, acute accent
+ { "\u00DE", "THORN" }, // Ю - uppercase THORN, Icelandic
+ { "\u00DF", "szlig" }, // Я - lowercase sharps, German
+ { "\u00E0", "agrave" }, // а - lowercase a, grave accent
+ { "\u00E1", "aacute" }, // б - lowercase a, acute accent
+ { "\u00E2", "acirc" }, // в - lowercase a, circumflex accent
+ { "\u00E3", "atilde" }, // г - lowercase a, tilde
+ { "\u00E4", "auml" }, // д - lowercase a, umlaut
+ { "\u00E5", "aring" }, // е - lowercase a, ring
+ { "\u00E6", "aelig" }, // ж - lowercase ae
+ { "\u00E7", "ccedil" }, // з - lowercase c, cedilla
+ { "\u00E8", "egrave" }, // и - lowercase e, grave accent
+ { "\u00E9", "eacute" }, // й - lowercase e, acute accent
+ { "\u00EA", "ecirc" }, // к - lowercase e, circumflex accent
+ { "\u00EB", "euml" }, // л - lowercase e, umlaut
+ { "\u00EC", "igrave" }, // м - lowercase i, grave accent
+ { "\u00ED", "iacute" }, // н - lowercase i, acute accent
+ { "\u00EE", "icirc" }, // о - lowercase i, circumflex accent
+ { "\u00EF", "iuml" }, // п - lowercase i, umlaut
+ { "\u00F0", "eth" }, // р - lowercase eth, Icelandic
+ { "\u00F1", "ntilde" }, // с - lowercase n, tilde
+ { "\u00F2", "ograve" }, // т - lowercase o, grave accent
+ { "\u00F3", "oacute" }, // у - lowercase o, acute accent
+ { "\u00F4", "ocirc" }, // ф - lowercase o, circumflex accent
+ { "\u00F5", "otilde" }, // х - lowercase o, tilde
+ { "\u00F6", "ouml" }, // ц - lowercase o, umlaut
+ { "\u00F7", "divide" }, // division sign
+ { "\u00F8", "oslash" }, // ш - lowercase o, slash
+ { "\u00F9", "ugrave" }, // щ - lowercase u, grave accent
+ { "\u00FA", "uacute" }, // ъ - lowercase u, acute accent
+ { "\u00FB", "ucirc" }, // ы - lowercase u, circumflex accent
+ { "\u00FC", "uuml" }, // ь - lowercase u, umlaut
+ { "\u00FD", "yacute" }, // э - lowercase y, acute accent
+ { "\u00FE", "thorn" }, // ю - lowercase thorn, Icelandic
+ { "\u00FF", "yuml" }, // я - lowercase y, umlaut
+ };
+
+ private static final int MIN_ESCAPE = 2;
+ private static final int MAX_ESCAPE = 6;
+
+ private static final HashMap<String, CharSequence> lookupMap;
+ static {
+ lookupMap = new HashMap<String, CharSequence>();
+ for (final CharSequence[] seq : ESCAPES)
+ lookupMap.put(seq[1].toString(), seq[0]);
+ }
+}
diff --git a/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackOverflowIdentityProviderConfig.java b/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackOverflowIdentityProviderConfig.java
new file mode 100644
index 0000000..f531d7c
--- /dev/null
+++ b/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackOverflowIdentityProviderConfig.java
@@ -0,0 +1,40 @@
+/*
+ * JBoss, Home of Professional Open Source
+ *
+ * Copyright 2013 Red Hat, Inc. and/or its affiliates.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.keycloak.social.stackoverflow;
+
+import org.keycloak.broker.oidc.OAuth2IdentityProviderConfig;
+import org.keycloak.models.IdentityProviderModel;
+
+/**
+ * @author Vlastimil Elias (velias at redhat dot com)
+ */
+public class StackOverflowIdentityProviderConfig extends OAuth2IdentityProviderConfig {
+
+ public StackOverflowIdentityProviderConfig(IdentityProviderModel model) {
+ super(model);
+ }
+
+ public String getKey() {
+ return getConfig().get("key");
+ }
+
+ public void setKey(String key) {
+ getConfig().put("key", key);
+ }
+
+}
\ No newline at end of file
diff --git a/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackoverflowIdentityProviderFactory.java b/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackoverflowIdentityProviderFactory.java
new file mode 100644
index 0000000..d02c2d7
--- /dev/null
+++ b/social/stackoverflow/src/main/java/org/keycloak/social/stackoverflow/StackoverflowIdentityProviderFactory.java
@@ -0,0 +1,47 @@
+/*
+ * JBoss, Home of Professional Open Source
+ *
+ * Copyright 2015 Red Hat, Inc. and/or its affiliates.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.keycloak.social.stackoverflow;
+
+import org.keycloak.broker.provider.AbstractIdentityProviderFactory;
+import org.keycloak.models.IdentityProviderModel;
+import org.keycloak.social.SocialIdentityProviderFactory;
+
+/**
+ * @author Vlastimil Elias (velias at redhat dot com)
+ */
+public class StackoverflowIdentityProviderFactory extends
+ AbstractIdentityProviderFactory<StackoverflowIdentityProvider> implements
+ SocialIdentityProviderFactory<StackoverflowIdentityProvider> {
+
+ public static final String PROVIDER_ID = "stackoverflow";
+
+ @Override
+ public String getName() {
+ return "StackOverflow";
+ }
+
+ @Override
+ public StackoverflowIdentityProvider create(IdentityProviderModel model) {
+ return new StackoverflowIdentityProvider(new StackOverflowIdentityProviderConfig(model));
+ }
+
+ @Override
+ public String getId() {
+ return PROVIDER_ID;
+ }
+}
diff --git a/social/stackoverflow/src/main/resources/META-INF/services/org.keycloak.social.SocialIdentityProviderFactory b/social/stackoverflow/src/main/resources/META-INF/services/org.keycloak.social.SocialIdentityProviderFactory
new file mode 100644
index 0000000..4bbbe9c
--- /dev/null
+++ b/social/stackoverflow/src/main/resources/META-INF/services/org.keycloak.social.SocialIdentityProviderFactory
@@ -0,0 +1 @@
+org.keycloak.social.stackoverflow.StackoverflowIdentityProviderFactory
\ No newline at end of file