// Copyright (c) 2016 Mattermost, Inc. All Rights Reserved. // See License.txt for license information. package app import ( "fmt" "net/http" "strconv" "strings" l4g "github.com/alecthomas/log4go" "github.com/mattermost/platform/model" "github.com/mattermost/platform/utils" ) func CreateTeam(team *model.Team) (*model.Team, *model.AppError) { if result := <-Srv.Store.Team().Save(team); result.Err != nil { return nil, result.Err } else { rteam := result.Data.(*model.Team) if _, err := CreateDefaultChannels(rteam.Id); err != nil { return nil, err } return rteam, nil } } func CreateTeamWithUser(team *model.Team, userId string) (*model.Team, *model.AppError) { var user *model.User var err *model.AppError if user, err = GetUser(userId); err != nil { return nil, err } else { team.Email = user.Email } if !isTeamEmailAllowed(user) { return nil, model.NewLocAppError("isTeamEmailAllowed", "api.team.is_team_creation_allowed.domain.app_error", nil, "") } var rteam *model.Team if rteam, err = CreateTeam(team); err != nil { return nil, err } if err = JoinUserToTeam(rteam, user); err != nil { return nil, err } return rteam, nil } func isTeamEmailAllowed(user *model.User) bool { email := strings.ToLower(user.Email) if len(user.AuthService) > 0 && len(*user.AuthData) > 0 { return true } // commas and @ signs are optional // can be in the form of "@corp.mattermost.com, mattermost.com mattermost.org" -> corp.mattermost.com mattermost.com mattermost.org domains := strings.Fields(strings.TrimSpace(strings.ToLower(strings.Replace(strings.Replace(utils.Cfg.TeamSettings.RestrictCreationToDomains, "@", " ", -1), ",", " ", -1)))) matched := false for _, d := range domains { if strings.HasSuffix(email, "@"+d) { matched = true break } } if len(utils.Cfg.TeamSettings.RestrictCreationToDomains) > 0 && !matched { return false } return true } func UpdateTeam(team *model.Team) (*model.Team, *model.AppError) { var oldTeam *model.Team var err *model.AppError if oldTeam, err = GetTeam(team.Id); err != nil { return nil, err } oldTeam.DisplayName = team.DisplayName oldTeam.Description = team.Description oldTeam.InviteId = team.InviteId oldTeam.AllowOpenInvite = team.AllowOpenInvite oldTeam.CompanyName = team.CompanyName oldTeam.AllowedDomains = team.AllowedDomains if result := <-Srv.Store.Team().Update(oldTeam); result.Err != nil { return nil, result.Err } oldTeam.Sanitize() message := model.NewWebSocketEvent(model.WEBSOCKET_EVENT_UPDATE_TEAM, "", "", "", nil) message.Add("team", oldTeam.ToJson()) go Publish(message) return oldTeam, nil } func UpdateTeamMemberRoles(teamId string, userId string, newRoles string) (*model.TeamMember, *model.AppError) { var member *model.TeamMember if result := <-Srv.Store.Team().GetTeamsForUser(userId); result.Err != nil { return nil, result.Err } else { members := result.Data.([]*model.TeamMember) for _, m := range members { if m.TeamId == teamId { member = m } } } if member == nil { err := model.NewLocAppError("UpdateTeamMemberRoles", "api.team.update_member_roles.not_a_member", nil, "userId="+userId+" teamId="+teamId) err.StatusCode = http.StatusBadRequest return nil, err } member.Roles = newRoles if result := <-Srv.Store.Team().UpdateMember(member); result.Err != nil { return nil, result.Err } ClearSessionCacheForUser(userId) return member, nil } func AddUserToTeam(teamId string, userId string) (*model.Team, *model.AppError) { tchan := Srv.Store.Team().Get(teamId) uchan := Srv.Store.User().Get(userId) var team *model.Team if result := <-tchan; result.Err != nil { return nil, result.Err } else { team = result.Data.(*model.Team) } var user *model.User if result := <-uchan; result.Err != nil { return nil, result.Err } else { user = result.Data.(*model.User) } if err := JoinUserToTeam(team, user); err != nil { return nil, err } return team, nil } func AddUserToTeamByTeamId(teamId string, user *model.User) *model.AppError { if result := <-Srv.Store.Team().Get(teamId); result.Err != nil { return result.Err } else { return JoinUserToTeam(result.Data.(*model.Team), user) } } func AddUserToTeamByHash(userId string, hash string, data string) (*model.Team, *model.AppError) { props := model.MapFromJson(strings.NewReader(data)) if !model.ComparePassword(hash, fmt.Sprintf("%v:%v", data, utils.Cfg.EmailSettings.InviteSalt)) { return nil, model.NewLocAppError("JoinUserToTeamByHash", "api.user.create_user.signup_link_invalid.app_error", nil, "") } t, timeErr := strconv.ParseInt(props["time"], 10, 64) if timeErr != nil || model.GetMillis()-t > 1000*60*60*48 { // 48 hours return nil, model.NewLocAppError("JoinUserToTeamByHash", "api.user.create_user.signup_link_expired.app_error", nil, "") } tchan := Srv.Store.Team().Get(props["id"]) uchan := Srv.Store.User().Get(userId) var team *model.Team if result := <-tchan; result.Err != nil { return nil, result.Err } else { team = result.Data.(*model.Team) } var user *model.User if result := <-uchan; result.Err != nil { return nil, result.Err } else { user = result.Data.(*model.User) } if err := JoinUserToTeam(team, user); err != nil { return nil, err } return team, nil } func AddUserToTeamByInviteId(inviteId string, userId string) (*model.Team, *model.AppError) { tchan := Srv.Store.Team().GetByInviteId(inviteId) uchan := Srv.Store.User().Get(userId) var team *model.Team if result := <-tchan; result.Err != nil { return nil, result.Err } else { team = result.Data.(*model.Team) } var user *model.User if result := <-uchan; result.Err != nil { return nil, result.Err } else { user = result.Data.(*model.User) } if err := JoinUserToTeam(team, user); err != nil { return nil, err } return team, nil } func JoinUserToTeam(team *model.Team, user *model.User) *model.AppError { tm := &model.TeamMember{ TeamId: team.Id, UserId: user.Id, Roles: model.ROLE_TEAM_USER.Id, } channelRole := model.ROLE_CHANNEL_USER.Id if team.Email == user.Email { tm.Roles = model.ROLE_TEAM_USER.Id + " " + model.ROLE_TEAM_ADMIN.Id channelRole = model.ROLE_CHANNEL_USER.Id + " " + model.ROLE_CHANNEL_ADMIN.Id } if etmr := <-Srv.Store.Team().GetMember(team.Id, user.Id); etmr.Err == nil { // Membership alredy exists. Check if deleted and and update, otherwise do nothing rtm := etmr.Data.(*model.TeamMember) // Do nothing if already added if rtm.DeleteAt == 0 { return nil } if tmr := <-Srv.Store.Team().UpdateMember(tm); tmr.Err != nil { return tmr.Err } } else { // Membership appears to be missing. Lets try to add. if tmr := <-Srv.Store.Team().SaveMember(tm); tmr.Err != nil { return tmr.Err } } if uua := <-Srv.Store.User().UpdateUpdateAt(user.Id); uua.Err != nil { return uua.Err } // Soft error if there is an issue joining the default channels if err := JoinDefaultChannels(team.Id, user, channelRole); err != nil { l4g.Error(utils.T("api.user.create_user.joining.error"), user.Id, team.Id, err) } ClearSessionCacheForUser(user.Id) InvalidateCacheForUser(user.Id) return nil } func GetTeam(teamId string) (*model.Team, *model.AppError) { if result := <-Srv.Store.Team().Get(teamId); result.Err != nil { return nil, result.Err } else { return result.Data.(*model.Team), nil } } func GetTeamByName(name string) (*model.Team, *model.AppError) { if result := <-Srv.Store.Team().GetByName(name); result.Err != nil { return nil, result.Err } else { return result.Data.(*model.Team), nil } } func GetTeamByInviteId(inviteId string) (*model.Team, *model.AppError) { if result := <-Srv.Store.Team().GetByInviteId(inviteId); result.Err != nil { return nil, result.Err } else { return result.Data.(*model.Team), nil } } func GetAllTeams() ([]*model.Team, *model.AppError) { if result := <-Srv.Store.Team().GetAll(); result.Err != nil { return nil, result.Err } else { return result.Data.([]*model.Team), nil } } func GetAllOpenTeams() ([]*model.Team, *model.AppError) { if result := <-Srv.Store.Team().GetAllTeamListing(); result.Err != nil { return nil, result.Err } else { return result.Data.([]*model.Team), nil } } func GetTeamsForUser(userId string) ([]*model.Team, *model.AppError) { if result := <-Srv.Store.Team().GetTeamsByUserId(userId); result.Err != nil { return nil, result.Err } else { return result.Data.([]*model.Team), nil } } func GetTeamMember(teamId, userId string) (*model.TeamMember, *model.AppError) { if result := <-Srv.Store.Team().GetMember(teamId, userId); result.Err != nil { return nil, result.Err } else { return result.Data.(*model.TeamMember), nil } } func GetTeamMembersForUser(userId string) ([]*model.TeamMember, *model.AppError) { if result := <-Srv.Store.Team().GetTeamsForUser(userId); result.Err != nil { return nil, result.Err } else { return result.Data.([]*model.TeamMember), nil } } func GetTeamMembers(teamId string, offset int, limit int) ([]*model.TeamMember, *model.AppError) { if result := <-Srv.Store.Team().GetMembers(teamId, offset, limit); result.Err != nil { return nil, result.Err } else { return result.Data.([]*model.TeamMember), nil } } func GetTeamMembersByIds(teamId string, userIds []string) ([]*model.TeamMember, *model.AppError) { if result := <-Srv.Store.Team().GetMembersByIds(teamId, userIds); result.Err != nil { return nil, result.Err } else { return result.Data.([]*model.TeamMember), nil } } func RemoveUserFromTeam(teamId string, userId string) *model.AppError { tchan := Srv.Store.Team().Get(teamId) uchan := Srv.Store.User().Get(userId) var team *model.Team if result := <-tchan; result.Err != nil { return result.Err } else { team = result.Data.(*model.Team) } var user *model.User if result := <-uchan; result.Err != nil { return result.Err } else { user = result.Data.(*model.User) } if err := LeaveTeam(team, user); err != nil { return err } return nil } func LeaveTeam(team *model.Team, user *model.User) *model.AppError { var teamMember *model.TeamMember var err *model.AppError if teamMember, err = GetTeamMember(team.Id, user.Id); err != nil { return model.NewLocAppError("LeaveTeam", "api.team.remove_user_from_team.missing.app_error", nil, err.Error()) } var channelList *model.ChannelList if result := <-Srv.Store.Channel().GetChannels(team.Id, user.Id); result.Err != nil { if result.Err.Id == "store.sql_channel.get_channels.not_found.app_error" { channelList = &model.ChannelList{} } else { return result.Err } } else { channelList = result.Data.(*model.ChannelList) } for _, channel := range *channelList { if channel.Type != model.CHANNEL_DIRECT { InvalidateCacheForChannelMembers(channel.Id) if result := <-Srv.Store.Channel().RemoveMember(channel.Id, user.Id); result.Err != nil { return result.Err } } } // Send the websocket message before we actually do the remove so the user being removed gets it. message := model.NewWebSocketEvent(model.WEBSOCKET_EVENT_LEAVE_TEAM, team.Id, "", "", nil) message.Add("user_id", user.Id) message.Add("team_id", team.Id) Publish(message) teamMember.Roles = "" teamMember.DeleteAt = model.GetMillis() if result := <-Srv.Store.Team().UpdateMember(teamMember); result.Err != nil { return result.Err } if uua := <-Srv.Store.User().UpdateUpdateAt(user.Id); uua.Err != nil { return uua.Err } // delete the preferences that set the last channel used in the team and other team specific preferences if result := <-Srv.Store.Preference().DeleteCategory(user.Id, team.Id); result.Err != nil { return result.Err } ClearSessionCacheForUser(user.Id) InvalidateCacheForUser(user.Id) return nil } func InviteNewUsersToTeam(emailList []string, teamId, senderId, siteURL string) *model.AppError { if len(emailList) == 0 { err := model.NewLocAppError("InviteNewUsersToTeam", "api.team.invite_members.no_one.app_error", nil, "") err.StatusCode = http.StatusBadRequest return err } tchan := Srv.Store.Team().Get(teamId) uchan := Srv.Store.User().Get(senderId) var team *model.Team if result := <-tchan; result.Err != nil { return result.Err } else { team = result.Data.(*model.Team) } var user *model.User if result := <-uchan; result.Err != nil { return result.Err } else { user = result.Data.(*model.User) } SendInviteEmails(team, user.GetDisplayName(), emailList, siteURL) return nil } func FindTeamByName(name string) bool { if result := <-Srv.Store.Team().GetByName(name); result.Err != nil { return false } else { return true } } func GetTeamsUnreadForUser(teamId string, userId string) ([]*model.TeamUnread, *model.AppError) { if result := <-Srv.Store.Team().GetTeamsUnreadForUser(teamId, userId); result.Err != nil { return nil, result.Err } else { data := result.Data.([]*model.ChannelUnread) var members []*model.TeamUnread membersMap := make(map[string]*model.TeamUnread) unreads := func(cu *model.ChannelUnread, tu *model.TeamUnread) *model.TeamUnread { tu.MentionCount += cu.MentionCount if cu.NotifyProps["mark_unread"] != model.CHANNEL_MARK_UNREAD_MENTION { tu.MsgCount += (cu.TotalMsgCount - cu.MsgCount) } return tu } for i := range data { id := data[i].TeamId if mu, ok := membersMap[id]; ok { membersMap[id] = unreads(data[i], mu) } else { membersMap[id] = unreads(data[i], &model.TeamUnread{ MsgCount: 0, MentionCount: 0, TeamId: id, }) } } for _, val := range membersMap { members = append(members, val) } return members, nil } } func PermanentDeleteTeam(team *model.Team) *model.AppError { team.DeleteAt = model.GetMillis() if result := <-Srv.Store.Team().Update(team); result.Err != nil { return result.Err } if result := <-Srv.Store.Channel().PermanentDeleteByTeam(team.Id); result.Err != nil { return result.Err } if result := <-Srv.Store.Team().RemoveAllMembersByTeam(team.Id); result.Err != nil { return result.Err } if result := <-Srv.Store.Team().PermanentDelete(team.Id); result.Err != nil { return result.Err } return nil } func GetTeamStats(teamId string) (*model.TeamStats, *model.AppError) { tchan := Srv.Store.Team().GetTotalMemberCount(teamId) achan := Srv.Store.Team().GetActiveMemberCount(teamId) stats := &model.TeamStats{} stats.TeamId = teamId if result := <-tchan; result.Err != nil { return nil, result.Err } else { stats.TotalMemberCount = result.Data.(int64) } if result := <-achan; result.Err != nil { return nil, result.Err } else { stats.ActiveMemberCount = result.Data.(int64) } return stats, nil }