
* datepicker-fr.js: datepicker french localization * jquery-2.1.3.js: * LocalizedText.resx: * jquery-2.1.1.min.js: * jquery-2.1.3.min.js: * jquery-2.1.1.min.map: * jquery-2.1.3.min.map: * jquery-2.1.1-vsdoc.js: * jquery-2.1.3-vsdoc.js: * globalize.js: * globalize.cultures.js: * globalize.culture.fy.js: * globalize.culture.fr.js: * globalize.culture.or.js: * globalize.culture.pa.js: * globalize.culture.pl.js: * globalize.culture.fo.js: * globalize.culture.ps.js: * globalize.culture.oc.js: * globalize.culture.nl.js: * globalize.culture.gu.js: * globalize.culture.nn.js: * globalize.culture.no.js: * globalize.culture.gl.js: * globalize.culture.gd.js: * globalize.culture.ga.js: * globalize.culture.fi.js: * globalize.culture.sa.js: * globalize.culture.se.js: * globalize.culture.si.js: * globalize.culture.sk.js: * globalize.culture.sl.js: * globalize.culture.sq.js: * globalize.culture.sr.js: * globalize.culture.rw.js: * globalize.culture.fa.js: * globalize.culture.eu.js: * globalize.culture.pt.js: * globalize.culture.et.js: * globalize.culture.es.js: * globalize.culture.rm.js: * globalize.culture.ro.js: * globalize.culture.ru.js: * globalize.culture.ha.js: * globalize.culture.kl.js: * globalize.culture.mi.js: * globalize.culture.kk.js: * globalize.culture.ka.js: * globalize.culture.mk.js: * globalize.culture.ja.js: * globalize.culture.iu.js: * globalize.culture.km.js: * globalize.culture.lb.js: * globalize.culture.lo.js: * globalize.culture.ky.js: * globalize.culture.lt.js: * globalize.culture.ko.js: * globalize.culture.kn.js: * globalize.culture.lv.js: * globalize.culture.ml.js: * globalize.culture.mt.js: * globalize.culture.hu.js: * globalize.culture.nb.js: * globalize.culture.hr.js: * globalize.culture.ne.js: * globalize.culture.hi.js: * globalize.culture.he.js: * globalize.culture.hy.js: * globalize.culture.mn.js: * globalize.culture.it.js: * globalize.culture.mr.js: * globalize.culture.is.js: * globalize.culture.ii.js: * globalize.culture.ig.js: * globalize.culture.ms.js: * globalize.culture.id.js: * globalize.culture.bg.js: * globalize.culture.de.js: * globalize.culture.co.js: * globalize.culture.tr.js: * globalize.culture.te.js: * globalize.culture.ar.js: * globalize.culture.ca.js: * globalize.culture.az.js: * globalize.culture.bs.js: * globalize.culture.tt.js: * globalize.culture.wo.js: * globalize.culture.xh.js: * globalize.culture.as.js: * globalize.culture.cy.js: * globalize.culture.da.js: * globalize.culture.tk.js: * globalize.culture.tn.js: * globalize.culture.cs.js: * globalize.culture.th.js: * globalize.culture.yo.js: * globalize.culture.tg.js: * globalize.culture.dv.js: * globalize.culture.be.js: * globalize.culture.sw.js: * globalize.culture.bn.js: * globalize.culture.uz.js: * globalize.culture.ur.js: * globalize.culture.zu.js: * globalize.culture.am.js: * globalize.culture.sv.js: * globalize.culture.af.js: * globalize.culture.uk.js: * globalize.culture.br.js: * globalize.culture.ta.js: * globalize.culture.vi.js: * globalize.culture.el.js: * globalize.culture.zh.js: * globalize.culture.ba.js: * globalize.culture.bo.js: * globalize.culture.ug.js: * globalize.culture.gsw.js: * globalize.culture.tzm.js: * globalize.culture.hsb.js: * globalize.culture.moh.js: * globalize.culture.prs.js: * globalize.culture.quz.js: * globalize.culture.dsb.js: * globalize.culture.sma.js: * globalize.culture.syr.js: * globalize.culture.kok.js: * globalize.culture.sah.js: * globalize.culture.qut.js: * globalize.culture.fil.js: * globalize.culture.sms.js: * globalize.culture.nso.js: * globalize.culture.smn.js: * globalize.culture.smj.js: * globalize.culture.arn.js: * globalize.culture.lv-LV.js: * globalize.culture.ms-MY.js: * globalize.culture.zu-ZA.js: * globalize.culture.vi-VN.js: * globalize.culture.mt-MT.js: * globalize.culture.lt-LT.js: * globalize.culture.zh-TW.js: * globalize.culture.mn-MN.js: * globalize.culture.xh-ZA.js: * globalize.culture.yo-NG.js: * globalize.culture.mk-MK.js: * globalize.culture.mr-IN.js: * globalize.culture.wo-SN.js: * globalize.culture.zh-SG.js: * globalize.culture.ms-BN.js: * globalize.culture.zh-MO.js: * globalize.culture.mi-NZ.js: * globalize.culture.zh-HK.js: * globalize.culture.zh-CN.js: * globalize.culture.ml-IN.js: * globalize.culture.sa-IN.js: * globalize.culture.rw-RW.js: * globalize.culture.te-IN.js: * globalize.culture.ru-RU.js: * globalize.culture.ro-RO.js: * globalize.culture.rm-CH.js: * globalize.culture.th-TH.js: * globalize.culture.pt-PT.js: * globalize.culture.se-FI.js: * globalize.culture.sv-FI.js: * globalize.culture.sw-KE.js: * globalize.culture.sq-AL.js: * globalize.culture.sl-SI.js: * globalize.culture.sk-SK.js: * globalize.culture.ta-IN.js: * globalize.culture.si-LK.js: * globalize.culture.se-SE.js: * globalize.culture.se-NO.js: * globalize.culture.tk-TM.js: * globalize.culture.ug-CN.js: * globalize.culture.nn-NO.js: * globalize.culture.nl-NL.js: * globalize.culture.uk-UA.js: * globalize.culture.nl-BE.js: * globalize.culture.ne-NP.js: * globalize.culture.ur-PK.js: * globalize.culture.nb-NO.js: * globalize.culture.sv-SE.js: * globalize.culture.pt-BR.js: * globalize.culture.ps-AF.js: * globalize.culture.tn-ZA.js: * globalize.culture.pl-PL.js: * globalize.culture.pa-IN.js: * globalize.culture.tr-TR.js: * globalize.culture.or-IN.js: * globalize.culture.oc-FR.js: * globalize.culture.tt-RU.js: * globalize.culture.lo-LA.js: * globalize.culture.en-IE.js: * globalize.culture.en-IN.js: * globalize.culture.en-JM.js: * globalize.culture.en-MY.js: * globalize.culture.en-NZ.js: * globalize.culture.en-PH.js: * globalize.culture.en-SG.js: * globalize.culture.en-GB.js: * globalize.culture.de-LI.js: * globalize.culture.de-LU.js: * globalize.culture.dv-MV.js: * globalize.culture.el-GR.js: * globalize.culture.en-AU.js: * globalize.culture.en-BZ.js: * globalize.culture.en-CA.js: * globalize.culture.en-TT.js: * globalize.culture.es-CR.js: * globalize.culture.es-DO.js: * globalize.culture.es-EC.js: * globalize.culture.es-ES.js: * globalize.culture.es-GT.js: * globalize.culture.es-HN.js: * globalize.culture.es-MX.js: * globalize.culture.es-CO.js: * globalize.culture.kn-IN.js: * globalize.culture.en-US.js: * globalize.culture.en-ZA.js: * globalize.culture.en-ZW.js: * globalize.culture.es-AR.js: * globalize.culture.es-BO.js: * globalize.culture.es-CL.js: * globalize.culture.de-DE.js: * globalize.culture.ar-LB.js: * globalize.culture.ar-LY.js: * globalize.culture.ar-MA.js: * globalize.culture.ar-OM.js: * globalize.culture.ar-QA.js: * globalize.culture.ar-SA.js: * globalize.culture.ar-SY.js: * globalize.culture.ar-KW.js: * globalize.culture.am-ET.js: * globalize.culture.ar-AE.js: * globalize.culture.ar-BH.js: * globalize.culture.ar-DZ.js: * globalize.culture.ar-EG.js: * globalize.culture.ar-IQ.js: * globalize.culture.ar-JO.js: * globalize.culture.ar-TN.js: * globalize.culture.ca-ES.js: * globalize.culture.co-FR.js: * globalize.culture.cs-CZ.js: * globalize.culture.cy-GB.js: * globalize.culture.da-DK.js: * globalize.culture.de-AT.js: * globalize.culture.de-CH.js: * globalize.culture.br-FR.js: * globalize.culture.ar-YE.js: * globalize.culture.as-IN.js: * globalize.culture.ba-RU.js: * globalize.culture.be-BY.js: * globalize.culture.bg-BG.js: * globalize.culture.bn-BD.js: * globalize.culture.bn-IN.js: * globalize.culture.bo-CN.js: * globalize.culture.es-NI.js: * globalize.culture.gu-IN.js: * globalize.culture.he-IL.js: * globalize.culture.hi-IN.js: * globalize.culture.hr-BA.js: * globalize.culture.hr-HR.js: * globalize.culture.hu-HU.js: * globalize.culture.gl-ES.js: * globalize.culture.fr-CH.js: * globalize.culture.fr-FR.js: * globalize.culture.fr-LU.js: * globalize.culture.fr-MC.js: * globalize.culture.fy-NL.js: * globalize.culture.ga-IE.js: * globalize.culture.gd-GB.js: * globalize.culture.hy-AM.js: * globalize.culture.kk-KZ.js: * globalize.culture.kl-GL.js: * globalize.culture.km-KH.js: * globalize.culture.af-ZA.js: * globalize.culture.ko-KR.js: * globalize.culture.ky-KG.js: * globalize.culture.lb-LU.js: * globalize.culture.ka-GE.js: * globalize.culture.id-ID.js: * globalize.culture.ig-NG.js: * globalize.culture.ii-CN.js: * globalize.culture.is-IS.js: * globalize.culture.it-CH.js: * globalize.culture.it-IT.js: * globalize.culture.ja-JP.js: * globalize.culture.fa-IR.js: * globalize.culture.es-UY.js: * globalize.culture.fr-CA.js: * globalize.culture.es-VE.js: * globalize.culture.fr-BE.js: * globalize.culture.eu-ES.js: * globalize.culture.fo-FO.js: * globalize.culture.et-EE.js: * globalize.culture.es-US.js: * globalize.culture.es-PA.js: * globalize.culture.es-PR.js: * globalize.culture.fi-FI.js: * globalize.culture.es-PY.js: * globalize.culture.es-SV.js: * globalize.culture.es-PE.js: * globalize.culture.qut-GT.js: * globalize.culture.hsb-DE.js: * globalize.culture.smj-SE.js: * globalize.culture.sma-SE.js: * globalize.culture.quz-BO.js: * globalize.culture.kok-IN.js: * globalize.culture.sah-RU.js: * globalize.culture.moh-CA.js: * globalize.culture.sma-NO.js: * globalize.culture.zh-CHS.js: * globalize.culture.arn-CL.js: * globalize.culture.quz-PE.js: * globalize.culture.quz-EC.js: * globalize.culture.zh-CHT.js: * globalize.culture.sms-FI.js: * globalize.culture.en-029.js: * globalize.culture.smn-FI.js: * globalize.culture.gsw-FR.js: * globalize.culture.nso-ZA.js: * globalize.culture.prs-AF.js: * globalize.culture.dsb-DE.js: * globalize.culture.syr-SY.js: * globalize.culture.smj-NO.js: * globalize.culture.fil-PH.js: * globalize.culture.az-Latn.js: * globalize.culture.az-Cyrl.js: * globalize.culture.iu-Latn.js: * globalize.culture.mn-Mong.js: * globalize.culture.zh-Hant.js: * globalize.culture.zh-Hans.js: * globalize.culture.iu-Cans.js: * globalize.culture.mn-Cyrl.js: * globalize.culture.ha-Latn.js: * globalize.culture.uz-Cyrl.js: * globalize.culture.sr-Cyrl.js: * globalize.culture.sr-Latn.js: * globalize.culture.bs-Latn.js: * globalize.culture.uz-Latn.js: * globalize.culture.tg-Cyrl.js: * globalize.culture.bs-Cyrl.js: * globalize.culture.tzm-Latn.js: * globalize.culture.sr-Latn-BA.js: * globalize.culture.sr-Latn-CS.js: * globalize.culture.sr-Cyrl-ME.js: * globalize.culture.sr-Latn-RS.js: * globalize.culture.sr-Cyrl-BA.js: * globalize.culture.sr-Cyrl-CS.js: * globalize.culture.sr-Latn-ME.js: * globalize.culture.bs-Latn-BA.js: * globalize.culture.bs-Cyrl-BA.js: * globalize.culture.ha-Latn-NG.js: * globalize.culture.uz-Cyrl-UZ.js: * globalize.culture.uz-Latn-UZ.js: * globalize.culture.tg-Cyrl-TJ.js: * globalize.culture.iu-Latn-CA.js: * globalize.culture.mn-Mong-CN.js: * globalize.culture.iu-Cans-CA.js: * globalize.culture.az-Cyrl-AZ.js: * globalize.culture.az-Latn-AZ.js: * globalize.culture.sr-Cyrl-RS.js: * globalize.culture.tzm-Latn-DZ.js: * Makefile: now deploy to yavsc main site * GoogleController.cs: fixes the bad request to google * App.master: fixes a bad html structure introducted in the last commit * style.css: input and field validation style * DateQuery.aspx: datepicker in french * Web.csproj: JQuery update * packages.config: using the Nuget Globalyze package * AskForADate.cs: Validation messages * LocalizedText.fr.resx: localizaton for the text "Date query" * jquery-2.1.1.js: jQuery update
469 lines
16 KiB
C#
469 lines
16 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Web;
|
|
using System.Threading;
|
|
using System.Web.Mvc;
|
|
using System.Configuration;
|
|
using System.Threading.Tasks;
|
|
using System.Text;
|
|
using Mono.Security.Protocol.Tls;
|
|
using System.Net;
|
|
using System.IO;
|
|
using Yavsc.Model;
|
|
using Newtonsoft.Json;
|
|
using Yavsc.Model.Google;
|
|
using Yavsc.Model.RolesAndMembers;
|
|
using System.Web.Security;
|
|
using System.Web.Profile;
|
|
|
|
namespace Yavsc.Controllers
|
|
{
|
|
|
|
public class GoogleController : Controller
|
|
{
|
|
// datetime format : yyyy-mm-ddTHH:MM:ss
|
|
// 2015-01-01T10:00:00-07:00
|
|
private string API_KEY="AIzaSyBV_LQHb22nGgjNvFzZwnQHjao3Q7IewRw";
|
|
|
|
private static string getPeopleUri = "https://www.googleapis.com/plus/v1/people";
|
|
private static string getCalListUri = "https://www.googleapis.com/calendar/v3/users/me/calendarList";
|
|
private static string getCalEntriesUri = "https://www.googleapis.com/calendar/v3/calendars/{0}/events";
|
|
|
|
private static string CLIENT_ID = "325408689282-6bekh7p3guj4k0f3301a6frf025cnrk1.apps.googleusercontent.com";
|
|
private static string CLIENT_SECRET = "MaxYcvJJCs2gDGvaELZbzwfL";
|
|
|
|
private static string[] SCOPES = {
|
|
"openid",
|
|
"profile",
|
|
"email"
|
|
};
|
|
|
|
private static string tokenUri = "https://accounts.google.com/o/oauth2/token";
|
|
private static string authUri = "https://accounts.google.com/o/oauth2/auth";
|
|
private static string dateFormat = "yyyy-MM-ddTHH:mm:ss";
|
|
private string timeZone = "+02:00";
|
|
|
|
private string SetSessionSate ()
|
|
{
|
|
Random rand = new Random ();
|
|
string state = "security_token" + rand.Next (100000).ToString () + rand.Next (100000).ToString ();
|
|
Session ["state"] = state;
|
|
return state;
|
|
}
|
|
|
|
public void Login (string returnUrl)
|
|
{
|
|
if (string.IsNullOrWhiteSpace (returnUrl))
|
|
returnUrl = "/";
|
|
Session ["returnUrl"] = returnUrl;
|
|
|
|
string redirectUri = Request.Url.Scheme + "://" + Request.Url.Authority + "/Google/Auth";
|
|
string scope = string.Join ("%20", SCOPES);
|
|
|
|
string prms = String.Format ("response_type=code&client_id={0}&redirect_uri={1}&scope={2}&state={3}&include_granted_scopes=false",
|
|
CLIENT_ID, redirectUri, scope, SetSessionSate ());
|
|
|
|
GetAuthResponse (prms);
|
|
}
|
|
|
|
private void GetAuthResponse (string prms)
|
|
{
|
|
string cont = null;
|
|
WebRequest wr = WebRequest.Create (authUri + "?" + prms);
|
|
wr.Method = "GET";
|
|
using (
|
|
WebResponse response = wr.GetResponse ()) {
|
|
string resQuery = response.ResponseUri.Query;
|
|
cont = HttpUtility.ParseQueryString (resQuery) ["continue"];
|
|
response.Close ();
|
|
|
|
}
|
|
wr.Abort ();
|
|
|
|
Response.Redirect (cont);
|
|
}
|
|
|
|
[Authorize]
|
|
public void GetCalAuth ()
|
|
{
|
|
string redirectUri = Request.Url.Scheme + "://" + Request.Url.Authority + "/Google/CalAuth";
|
|
string scope = string.Join ("%20", SCOPES);
|
|
scope += "%20https://www.googleapis.com/auth/calendar";
|
|
string prms = String.Format ("response_type=code&client_id={0}&redirect_uri={1}&scope={2}&state={3}&include_granted_scopes=false&access_type=offline",
|
|
CLIENT_ID, redirectUri, scope, SetSessionSate ());
|
|
Session ["calasked"] = true;
|
|
GetAuthResponse (prms);
|
|
}
|
|
|
|
[HttpGet]
|
|
[Authorize]
|
|
public ActionResult CalAuth ()
|
|
{
|
|
string redirectUri = Request.Url.Scheme + "://" + Request.Url.Authority + "/Google/CalAuth";
|
|
AuthToken gat = GetToken (TokenPostDataFromCode(redirectUri, GetCodeFromRequest()));
|
|
if (gat == null) {
|
|
return View ("Auth");
|
|
}
|
|
SaveToken (gat);
|
|
HttpContext.Profile.SetPropertyValue ("gcalapi", true);
|
|
string returnUrl = (string)Session ["returnUrl"];
|
|
Session ["returnUrl"]=null;
|
|
return Redirect (returnUrl);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Saves the token.
|
|
/// This calls the Profile.Save() method.
|
|
/// It should be called immediatly after getting the token from Google, in
|
|
/// order to save a descent value as expiration date.
|
|
/// </summary>
|
|
/// <param name="gat">Gat.</param>
|
|
private void SaveToken(AuthToken gat)
|
|
{
|
|
HttpContext.Profile.SetPropertyValue ("gtoken", gat.access_token);
|
|
if (gat.refresh_token!=null)
|
|
HttpContext.Profile.SetPropertyValue ("grefreshtoken", gat.refresh_token);
|
|
HttpContext.Profile.SetPropertyValue ("gtokentype", gat.token_type);
|
|
HttpContext.Profile.SetPropertyValue ("gtokenexpir", DateTime.Now.AddSeconds(gat.expires_in));
|
|
HttpContext.Profile.Save ();
|
|
}
|
|
|
|
private string GetCodeFromRequest()
|
|
{
|
|
string code = Request.Params ["code"];
|
|
string error = Request.Params ["error"];
|
|
if (error != null) {
|
|
ViewData ["Message"] =
|
|
string.Format (LocalizedText.Google_error,
|
|
LocalizedText.ResourceManager.GetString (error));
|
|
return null;
|
|
}
|
|
string state = Request.Params ["state"];
|
|
if (state != null && string.Compare ((string)Session ["state"], state) != 0) {
|
|
ViewData ["Message"] =
|
|
LocalizedText.ResourceManager.GetString ("invalid request state");
|
|
return null;
|
|
}
|
|
return code;
|
|
}
|
|
|
|
private AuthToken GetToken (string postdata)
|
|
{
|
|
Byte[] bytes = System.Text.Encoding.UTF8.GetBytes (postdata);
|
|
HttpWebRequest webreq = WebRequest.CreateHttp (tokenUri);
|
|
webreq.KeepAlive = false;
|
|
webreq.Timeout = 5000;
|
|
webreq.Proxy = null;
|
|
// Not implemented: webreq.ServicePoint.ConnectionLeaseTimeout = 5000;
|
|
webreq.ServicePoint.MaxIdleTime = 5000;
|
|
webreq.Method = "POST";
|
|
webreq.Accept = "application/json";
|
|
webreq.ContentType = "application/x-www-form-urlencoded";
|
|
webreq.ContentLength = bytes.Length;
|
|
|
|
using (Stream dataStream = webreq.GetRequestStream ()) {
|
|
dataStream.Write (bytes, 0, bytes.Length);
|
|
dataStream.Close ();
|
|
}
|
|
AuthToken gat =null;
|
|
|
|
using (WebResponse response = webreq.GetResponse ()) {
|
|
using (Stream responseStream = response.GetResponseStream ()) {
|
|
using (StreamReader readStream = new StreamReader (responseStream, Encoding.UTF8)) {
|
|
string responseStr = readStream.ReadToEnd ();
|
|
gat = JsonConvert.DeserializeObject<AuthToken> (responseStr);
|
|
readStream.Close ();
|
|
}
|
|
responseStream.Close ();
|
|
}
|
|
response.Close();
|
|
}
|
|
webreq.Abort ();
|
|
|
|
return gat;
|
|
}
|
|
|
|
private string TokenPostDataFromCode(string redirectUri, string code)
|
|
{
|
|
string postdata =
|
|
string.Format (
|
|
"redirect_uri={0}&client_id={1}&client_secret={2}&code={3}&grant_type=authorization_code",
|
|
HttpUtility.UrlEncode (redirectUri),
|
|
HttpUtility.UrlEncode (CLIENT_ID),
|
|
HttpUtility.UrlEncode (CLIENT_SECRET),
|
|
HttpUtility.UrlEncode (code));
|
|
return postdata;
|
|
}
|
|
|
|
[HttpGet]
|
|
public ActionResult Auth ()
|
|
{
|
|
string redirectUri = Request.Url.Scheme + "://" + Request.Url.Authority + "/Google/Auth";
|
|
AuthToken gat = GetToken (TokenPostDataFromCode( redirectUri, GetCodeFromRequest()));
|
|
if (gat == null) {
|
|
return View ();
|
|
}
|
|
string returnUrl = (string)Session ["returnUrl"];
|
|
|
|
SignIn regmod = new SignIn ();
|
|
HttpWebRequest webreppro = WebRequest.CreateHttp (getPeopleUri + "/me");
|
|
webreppro.ContentType = "application/http";
|
|
webreppro.Headers.Add (HttpRequestHeader.Authorization, gat.token_type + " " + gat.access_token);
|
|
webreppro.Method = "GET";
|
|
using (WebResponse proresp = webreppro.GetResponse ()) {
|
|
using (Stream prresponseStream = proresp.GetResponseStream ()) {
|
|
using (StreamReader readproresp = new StreamReader (prresponseStream, Encoding.UTF8)) {
|
|
string prresponseStr = readproresp.ReadToEnd ();
|
|
People me = JsonConvert.DeserializeObject<People> (prresponseStr);
|
|
// TODO use me.id to retreive an existing user
|
|
string accEmail = me.emails.Where (x => x.type == "account").First ().value;
|
|
MembershipUserCollection mbrs = Membership.FindUsersByEmail (accEmail);
|
|
if (mbrs.Count == 1) {
|
|
// TODO check the google id
|
|
// just set this user as logged on
|
|
FormsAuthentication.SetAuthCookie (me.displayName, true);
|
|
Session ["returnUrl"] = null;
|
|
prresponseStream.Close ();
|
|
proresp.Close ();
|
|
webreppro.Abort ();
|
|
return Redirect (returnUrl);
|
|
}
|
|
// else create the account
|
|
regmod.Email = accEmail;
|
|
regmod.UserName = me.displayName;
|
|
Session ["me"] = me;
|
|
Session ["GoogleAuthToken"] = gat;
|
|
webreppro.Abort ();
|
|
|
|
|
|
}
|
|
prresponseStream.Close ();
|
|
}
|
|
proresp.Close ();
|
|
}
|
|
webreppro.Abort ();
|
|
return Auth (regmod);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates an account using the Google authentification.
|
|
/// </summary>
|
|
/// <param name="regmod">Regmod.</param>
|
|
[HttpPost]
|
|
public ActionResult Auth (SignIn regmod)
|
|
{
|
|
if (ModelState.IsValid) {
|
|
if (Membership.GetUser (regmod.UserName) != null) {
|
|
ModelState.AddModelError ("UserName", "This user name already is in use");
|
|
return View ();
|
|
}
|
|
string returnUrl = (string)Session ["returnUrl"];
|
|
AuthToken gat = (AuthToken)Session ["GoogleAuthToken"];
|
|
People me = (People)Session ["me"];
|
|
if (gat == null || me == null)
|
|
throw new InvalidDataException ();
|
|
|
|
Random rand = new Random ();
|
|
string passwd = rand.Next (100000).ToString () + rand.Next (100000).ToString ();
|
|
|
|
MembershipCreateStatus mcs;
|
|
Membership.CreateUser (
|
|
regmod.UserName,
|
|
passwd,
|
|
regmod.Email,
|
|
null,
|
|
null,
|
|
true,
|
|
out mcs);
|
|
switch (mcs) {
|
|
case MembershipCreateStatus.DuplicateEmail:
|
|
ModelState.AddModelError ("Email", "Cette adresse e-mail correspond " +
|
|
"à un compte utilisateur existant");
|
|
return View (regmod);
|
|
case MembershipCreateStatus.DuplicateUserName:
|
|
ModelState.AddModelError ("UserName", "Ce nom d'utilisateur est " +
|
|
"déjà enregistré");
|
|
return View (regmod);
|
|
case MembershipCreateStatus.Success:
|
|
Membership.ValidateUser (regmod.UserName, passwd);
|
|
FormsAuthentication.SetAuthCookie (regmod.UserName, true);
|
|
|
|
HttpContext.Profile.Initialize (regmod.UserName, true);
|
|
HttpContext.Profile.SetPropertyValue ("Name", me.displayName);
|
|
// TODO use image
|
|
if (me.image != null) {
|
|
HttpContext.Profile.SetPropertyValue ("avatar", me.image.url);
|
|
}
|
|
if (me.placesLived != null) {
|
|
People.Place pplace = me.placesLived.Where (x => x.primary).First ();
|
|
if (pplace != null)
|
|
HttpContext.Profile.SetPropertyValue ("CityAndState", pplace.value);
|
|
}
|
|
if (me.url != null)
|
|
HttpContext.Profile.SetPropertyValue ("WebSite", me.url);
|
|
SaveToken (gat);
|
|
// already done in SaveToken: HttpContext.Profile.Save ();
|
|
return Redirect (returnUrl);
|
|
}
|
|
ViewData ["returnUrl"] = returnUrl;
|
|
}
|
|
return View (regmod);
|
|
}
|
|
|
|
private string GetFreshGoogleCredential (ProfileBase pr)
|
|
{
|
|
string token = (string) pr.GetPropertyValue ("gtoken");
|
|
string token_type = (string) pr.GetPropertyValue ("gtokentype");
|
|
DateTime token_exp = (DateTime) pr.GetPropertyValue ("gtokenexpir");
|
|
if (token_exp < DateTime.Now) {
|
|
string refresh_token = (string) pr.GetPropertyValue ("grefreshtoken");
|
|
AuthToken gat = GetToken(
|
|
string.Format("grant_type=refresh_token&client_id={0}&client_secret={1}&refresh_token={2}",
|
|
CLIENT_ID, CLIENT_SECRET, refresh_token));
|
|
token = gat.access_token;
|
|
pr.SetPropertyValue ("gtoken", token);
|
|
pr.Save ();
|
|
// assert gat.token_type == token_type
|
|
}
|
|
return token_type + " " + token;
|
|
}
|
|
|
|
[Authorize]
|
|
[HttpGet]
|
|
public ActionResult ChooseCalendar (string returnUrl)
|
|
{
|
|
Session ["ChooseCalReturnUrl"] = returnUrl;
|
|
bool hasCalAuth = (bool)HttpContext.Profile.GetPropertyValue ("gcalapi");
|
|
if (!hasCalAuth) {
|
|
Session["returnUrl"] = Request.Url.Scheme + "://" + Request.Url.Authority + "/Google/ChooseCalendar";
|
|
return RedirectToAction ("GetCalAuth");
|
|
}
|
|
|
|
string cred = GetFreshGoogleCredential (HttpContext.Profile);
|
|
|
|
HttpWebRequest webreq = WebRequest.CreateHttp (getCalListUri);
|
|
webreq.Headers.Add (HttpRequestHeader.Authorization, cred);
|
|
webreq.Method = "GET";
|
|
webreq.ContentType = "application/http";
|
|
CalendarList res = null;
|
|
using (WebResponse resp = webreq.GetResponse ()) {
|
|
using (Stream respstream = resp.GetResponseStream ()) {
|
|
using (StreamReader readresp = new StreamReader (respstream, Encoding.UTF8)) {
|
|
string responseStr = readresp.ReadToEnd ();
|
|
res = JsonConvert.DeserializeObject<CalendarList> (responseStr);
|
|
ViewData ["json"] = responseStr;
|
|
}
|
|
}
|
|
resp.Close ();
|
|
}
|
|
webreq.Abort ();
|
|
return View (res);
|
|
}
|
|
|
|
[HttpPost]
|
|
[Authorize]
|
|
public ActionResult SetCalendar (string calchoice)
|
|
{
|
|
HttpContext.Profile.SetPropertyValue ("gcalid", calchoice);
|
|
HttpContext.Profile.Save ();
|
|
|
|
string returnUrl = (string) Session ["ChooseCalReturnUrl"];
|
|
if (returnUrl != null) {
|
|
Session ["ChooseCalReturnUrl"] = null;
|
|
return Redirect (returnUrl);
|
|
}
|
|
return Redirect ("/");
|
|
}
|
|
|
|
[Authorize]
|
|
[HttpGet]
|
|
public ActionResult DateQuery()
|
|
{
|
|
return View (new AskForADate ());
|
|
}
|
|
|
|
[Authorize]
|
|
[HttpPost]
|
|
public ActionResult DateQuery(AskForADate model)
|
|
{
|
|
if (ModelState.IsValid) {
|
|
if (model.MinDate < DateTime.Now) {
|
|
ModelState.AddModelError ("MinTime", "This first date must be in the future.");
|
|
return View (model);
|
|
}
|
|
if (model.MinDate > model.MaxDate) {
|
|
ModelState.AddModelError ("MinTime", "This first date must be lower than the second one.");
|
|
return View (model);
|
|
}
|
|
var muc = Membership.FindUsersByName (model.UserName);
|
|
if (muc.Count==0) {
|
|
ModelState.AddModelError ("UserName", "Non existent user");
|
|
return View (model);
|
|
}
|
|
ProfileBase upr = ProfileBase.Create (model.UserName);
|
|
|
|
string calid = (string) upr.GetPropertyValue ("gcalid");
|
|
if (string.IsNullOrWhiteSpace(calid)) {
|
|
ModelState.AddModelError ("UserName", "L'utilisateur n'a pas de calendrier Google associé.");
|
|
return View (model);
|
|
}
|
|
|
|
DateTime mindate = model.MinDate;
|
|
mindate = mindate.AddHours (int.Parse (model.MinTime.Substring (0, 2)));
|
|
mindate = mindate.AddMinutes (int.Parse (model.MinTime.Substring (3, 2)));
|
|
DateTime maxdate = model.MaxDate;
|
|
maxdate = maxdate.AddHours (int.Parse (model.MaxTime.Substring (0, 2)));
|
|
maxdate = maxdate.AddMinutes (int.Parse (model.MaxTime.Substring (3, 2)));
|
|
string uri = string.Format (
|
|
getCalEntriesUri, HttpUtility.UrlEncode(calid)) +
|
|
string.Format ("?orderBy=startTime&singleEvents=true&timeMin={0}&timeMax={1}&key="+API_KEY,
|
|
HttpUtility.UrlEncode(mindate.ToString (dateFormat)+timeZone) ,
|
|
HttpUtility.UrlEncode(maxdate.ToString (dateFormat)+timeZone) );
|
|
|
|
HttpWebRequest webreq = WebRequest.CreateHttp (uri);
|
|
string cred = GetFreshGoogleCredential (upr);
|
|
webreq.Headers.Add (HttpRequestHeader.Authorization, cred);
|
|
webreq.Method = "GET";
|
|
webreq.ContentType = "application/http";
|
|
CalendarEntryList res = null;
|
|
try {
|
|
using (WebResponse resp = webreq.GetResponse ()) {
|
|
using (Stream respstream = resp.GetResponseStream ()) {
|
|
using (StreamReader readresp = new StreamReader (respstream, Encoding.UTF8)) {
|
|
string responseStr = readresp.ReadToEnd ();
|
|
try {
|
|
ViewData ["json"] = responseStr;
|
|
res = JsonConvert.DeserializeObject<CalendarEntryList> (responseStr);
|
|
}
|
|
catch (JsonReaderException ex) {
|
|
respstream.Close ();
|
|
resp.Close ();
|
|
webreq.Abort ();
|
|
return View ("JsonReaderError", new JsonReaderError() {Text= responseStr, Excepx = ex});
|
|
}
|
|
}
|
|
respstream.Close ();
|
|
}
|
|
resp.Close ();
|
|
}
|
|
}
|
|
catch (WebException ex)
|
|
{
|
|
string message;
|
|
using (var stream = ex.Response.GetResponseStream())
|
|
using (var reader = new StreamReader(stream))
|
|
{
|
|
message = reader.ReadToEnd();
|
|
}
|
|
webreq.Abort ();
|
|
return View ("JsonReaderError", new JsonReaderError() {Text= message, Excepx = ex});
|
|
|
|
}
|
|
webreq.Abort ();
|
|
return View (res);
|
|
}
|
|
return View (model);
|
|
}
|
|
}
|
|
} |