- Document
- 관리자 가이드
- 로그 설정
- License 표시 변경
- 한글팩 제작
- 한글 설치 프로그램 제작
- 테마 구조
- Dashlet
- SOAP/REST
- SugarCRM 아키텍처
- 사용 라이브러리
- 프로그램 호출 구조
- 디렉토리 구조
- 프로세스 구조
- 화면 구조
- Shortcut Menu
- 동적 구조
- 문제 해결
- Localization
- 한글팩 작성
- 주소
- 한글로 모듈(거래처) 목록 검색
- 메일 발송시 보낸 사람 이름
- Tips
- 참고 문헌
- 지원 업체
오픈소스로 제공되는, On-Premise/On-Demand/Appliance 방식의 CRM인 sugarCRM (Sugar Community Edition)을 소개한다.
홈페이지
다운로드 : http://sourceforge.net/project/showfiles.php?group_id=107819
라이센스 : AGPL v3 + 상용 라이센스
플랫폼 : Windows, Solaris, Linux, Mac OS
Document
기술지원
[http://apidocs.sugarcrm.com/ API Document] : phpDocumentor 2를 사용하여 추출한 API Document
- Ultimate : [http://apidocs.sugarcrm.com/api/6.5.4/ult API], Schema
- Enterprise : [http://apidocs.sugarcrm.com/api/6.5.4/ent API], Schema
- Corporate : [http://apidocs.sugarcrm.com/api/6.5.4/corp API], Schema
- Professional : [http://apidocs.sugarcrm.com/api/6.5.4/pro API], Schema
- Community : [http://apidocs.sugarcrm.com/api/6.5.4/ce API], Schema
Sugar Enterprise 6.5.0 Administration Guide - SugarCRM Support Site
관리자 가이드
로그 설정
로그 파일
- $SUGAR_HOME/include/SugarLogger/LoggerManager.php
- $SUGAR_HOME/include/SugarLogger/SugarLogger.php
$SUGAR_HOME/config.php에서 로그 설정
log_dir : 로그가 저장되는 폴더 log_file : 로그 파일명, 초기값은 "sugarcrm.log" log_memory_usage : logger => level : 로그 레벨 (debug, info, error, fatal, security, off) logger => file :ext : 로그 파일 확장자, 초기값은 ".log" :name : 로그 파일명, 초기값은 "sugarcrm" :dateFormat : 날자 포맷 :maxSize : 로그 파일 최대 크기 :maxLogs : 로그 파일 최대 갯수 :suffix : 로그 파일 뒤에 붙는 문자열, 초기값은 "%m_%Y"
프로그램에서 로그 남기기
$GLOBALS['log']('log'.md)->debug('로그 메시지');
$GLOBALS['log']('log'.md)->info('로그 메시지');
License 표시 변경
vi include/MVC/View/SugarView.php
- 666 라인에 있는 $copyright 를 변경 한다.
한글팩 제작
SugarCRM의 한글팩을 만들기 위해 en_us 언어로 부터 ko_kr 파일을 추출하여 한글 언어팩을 작성 한다.
- 한글팩 작성을 위한 폴더 생성
mkdir /tmp/sugar #--- 다운로드 받은 SugarCE-6.1.1.zip의 압축을 여기에 푼다.
mkdir /tmp/sugar_ko_kr
ant를 사용하여 언어로 부터 한글 파일을 복사하기 위한 build 파일을 만든다.
- vi /tmp/sugar_ko_kr/build.xml
- 영어 파일을 복사하여 한글 파일로 이름만 변경 한다. (위에서 만든 build.xml 파일 사용)
cd /tmp/sugar_ko_kr
ant init
- vi /tmp/sugar_ko_kr/manifest.php
array (
'exact_matches' => array(),
'regex_matches' => array (
0 => "6\.6\.1.*"
),
),
'acceptable_sugar_flavors' => array (
0 => 'CE',
1 => 'PRO',
2 => 'ENT',
),
'name' => 'Korean Language Pack',
'description' => 'Korean Language Pack',
'author' => 'pnuskgh',
'published_date' => '2011/01/23',
'version' => '0.1',
'type' => 'langpack',
'icon' => '',''
'is_uninstallable' => true,
);
$installdefs = array(
'id'=> 'langpack_ko_kr',
'copy' => array(
array('from'=> '',
'to'=> '.',
),
),
);
?>
SugarCRM 한글팩(langpack_ko_kr.zip) 작성
- /tmp/sugar_ko_kr/ 폴더에 있는 파일을 열어 열심히 한글로 번역 한다.
- /tmp/sugar_ko_kr/ 폴더의 내용을 묶어 langpack_ko_kr.zip 파일을 작성 한다.
한글 설치 프로그램 제작
- vi /install.php 파일을 다음과 같이 수정 한다.
$supportedLanguages = array(
'ko_KR' => 'Korea - 한국어', //--- 이 라인을 추가 한다.
'en_us' => 'English (US)',
$default_lang = 'ko_KR'; //--- 디폴트로 ko_KR을 지정 한다.
- /install/language/ko_KR.lang.php 파일을 작성 한다.
- /install/demoData.ko_KR.php 데모 데이터 파일을 작성 한다.
테마 구조
Dashlet
참고 문헌
SOAP/REST
wsdl2java.bat -o . -p com.sugar [http://cloud.smartprocess.co.kr/testEnt/service/v4/soap.php?wsdl](http://cloud.smartprocess.co.kr/testEnt/service/v4/soap.php?wsdl)
- SugarsoapLocatior.java
// private java.lang.String sugarsoapPort_address = "[http://cloud.smartprocess.co.kr/testEnt/service/v4/soap.php](http://cloud.smartprocess.co.kr/testEnt/service/v4/soap.php)";
private java.lang.String sugarsoapPort_address = Config.getString("sugar.serverurl", "[http://cloud.smartprocess.co.kr/testEnt/](http://cloud.smartprocess.co.kr/testEnt/)") + "service/v4/soap.php";
- SugarsoapBuindingStub.java
public static String SugarServerurl = Config.getString("sugar.serverurl", "[http://cloud.smartprocess.co.kr/testEnt/](http://cloud.smartprocess.co.kr/testEnt/)") + "service/v4/soap.php";
_call.setSOAPActionURI(SugarsoapBindingStub.SugarServerurl + "/login");
- SugarsoapBuindingStub.java 함수
public com.sugar.Entry_value login(com.sugar.User_auth user_auth, java.lang.String application_name, com.sugar.Name_value[](.md) name_value_list)
public void logout(java.lang.String session)
public com.sugar.Get_entry_result_version2 get_entry(java.lang.String session, java.lang.String module_name, java.lang.String id,
java.lang.String[] select_fields, com.sugar.Link_name_to_fields_array[](.md) link_name_to_fields_array, boolean track_view)
public com.sugar.Get_entry_result_version2 get_entries(java.lang.String session, java.lang.String module_name, java.lang.String[](.md) ids,
java.lang.String[] select_fields, com.sugar.Link_name_to_fields_array[](.md) link_name_to_fields_array, boolean track_view)
public com.sugar.Get_entry_list_result_version2 get_entry_list(java.lang.String session, java.lang.String module_name, java.lang.String query,
java.lang.String order_by, int offset, java.lang.String[] select_fields, com.sugar.Link_name_to_fields_array[](.md) link_name_to_fields_array,
int max_results, int deleted, boolean favorites)
public com.sugar.New_set_relationship_list_result set_relationship(java.lang.String session, java.lang.String module_name,
java.lang.String module_id, java.lang.String link_field_name, java.lang.String[] related_ids, com.sugar.Name_value[](.md) name_value_list, int delete)
public com.sugar.New_set_relationship_list_result set_relationships(java.lang.String session, java.lang.String[](.md) module_names,
java.lang.String[] module_ids, java.lang.String[] link_field_names, java.lang.String[][] related_ids, com.sugar.Name_value[][] name_value_lists, int[](.md) delete_array)
public com.sugar.Get_entry_result_version2 get_relationships(java.lang.String session, java.lang.String module_name, java.lang.String module_id,
java.lang.String link_field_name, java.lang.String related_module_query, java.lang.String[](.md) related_fields,
com.sugar.Link_name_to_fields_array[](.md) related_module_link_name_to_fields_array, int deleted, java.lang.String order_by)
public com.sugar.New_set_entry_result set_entry(java.lang.String session, java.lang.String module_name, com.sugar.Name_value[](.md) name_value_list)
public com.sugar.New_set_entries_result set_entries(java.lang.String session, java.lang.String module_name, com.sugar.Name_value[][](.md) name_value_lists)
public com.sugar.Get_server_info_result get_server_info()
public java.lang.String get_user_id(java.lang.String session)
public com.sugar.New_module_fields get_module_fields(java.lang.String session, java.lang.String module_name, java.lang.String[](.md) fields)
public int seamless_login(java.lang.String session)
public com.sugar.New_set_entry_result set_note_attachment(java.lang.String session, com.sugar.New_note_attachment note)
public com.sugar.New_return_note_attachment get_note_attachment(java.lang.String session, java.lang.String id)
public com.sugar.New_set_entry_result set_document_revision(java.lang.String session, com.sugar.Document_revision note)
public com.sugar.New_return_document_revision get_document_revision(java.lang.String session, java.lang.String i)
public com.sugar.Return_search_result search_by_module(java.lang.String session, java.lang.String search_string, java.lang.String[](.md) modules,
int offset, int max_results, java.lang.String assigned_user_id, java.lang.String[](.md) select_fields, boolean unified_search_only, boolean favorites)
public com.sugar.Module_list get_available_modules(java.lang.String session, java.lang.String filter)
public java.lang.String get_user_team_id(java.lang.String session)
public void set_campaign_merge(java.lang.String session, java.lang.String[](.md) targets, java.lang.String campaign_id)
public com.sugar.Get_entries_count_result get_entries_count(java.lang.String session, java.lang.String module_name, java.lang.String query, int deleted)
public com.sugar.Get_entry_result_for_reports get_report_entries(java.lang.String session, java.lang.String[] ids, java.lang.String[](.md) select_fields)
public java.lang.String[] get_module_fields_md5(java.lang.String session, java.lang.String[](.md) module_names)
public com.sugar.Last_viewed_entry[] get_last_viewed(java.lang.String session, java.lang.String[](.md) module_names)
public com.sugar.Upcoming_activity_entry[](.md) get_upcoming_activities(java.lang.String session)
SugarCRM 아키텍처
사용 라이브러리
{| cellspacing="1" cellpadding="1" border="1" width="100%" |- | width="30%" bgcolor="cyan" align="center" valign="middle" | 라이브러리 | width="20%" bgcolor="cyan" align="center" valign="middle" | 라이선스
| width="50%" bgcolor="cyan" align="center" valign="middle" | 상세 설명 |- | Sugar Community Edition | align="center" | GNU GPL 3.0 | 오픈소스 CRM |- | tcpdf | align="center" | GNU LGPL 2.1 | PDF 제작 |- | phpmailer | align="center" | GNU LGPL 2.1 | 메일 발송 |- | [PHP_Compat](PHP Compat.md) | align="center" | PHP License 3.0 | |- | nusoap | align="center" | Free | Web Services Toolkit for PHP |- | [HTTP_WebDAV_Server](HTTP WebDAV Server.md) | align="center" | PHP License 3.0 | WebDAV 서버 |- | [HTML_Safe](HTML Safe.md) | align="center" | BSD style license | |- | [domit_rss](Domit rss.md) | align="center" | Free | DOM based RSS parser for PHP |- | domit | align="center" | Free | DOM parser for PHP |- | [Crypt_Blowfish](Crypt Blowfish.md) | align="center" | PHP License 3.0 | |- | TreeView | align="center" | BSD License | Javascript, Tree 형태의 항목을 표시 |- | yui (Yahoo User Interface) | align="center" | BSD License | Javascript, UI 모듈 |- | [tiny_mce](Tiny mce.md) | align="center" | GNU LGPL 2.1 | Javascript, |- | [Ext JS](Ext JS.md) | align="center" | GNU LGPL 3.0 | Javascript, |- | jscalendar | align="center" | GNU LGPL 3.0 | Javascript, 달력 |}
프로그램 호출 구조
호출 URL: http://localhost/sugar/index.php?module=Accounts&action=index
- '''module''': /sugar/modules/Accounts/ 폴더에 있는 모듈이 사용됨
- '''action'''::*/sugar/include/MVC/Controller/action_file_map.php 참조
- /sugar/include/MVC/Controller/action_view_map.php 참조
- 아래 정리된 범례. action_명 (view_명)
- index:
- DetailView (detail):
- EditView (edit):
- MultiEditView (multiedit):
- Popup (popup):
- Vcard (vcard):
- ImportVcard (importvcard):
- ImportVcardSave (importvcardsave):
- SugarPdf (sugarpdf):
- SaveTimezone
- InitialSync
- Logout
- RetrieveEmail
- view 명: ajax, classic, config, detail, edit, html, importvcard, importvcardsave, json, list, multedit, noaccess, popup, serialized, sidequickcreate, sugarpdf, vcard, xml
- '''return_module'''
- '''return_action'''
- return_id:
- id:
- record:
- view:
- Delete:
- entire:
- mass:
- type:
- ie_assigned_user_id:
- configure:
- RTL:
- LTR:
- query_string:
- module_tab:
- parentTab:
- popup:
- lvso:
- query:
- displayColumns:
- orderBy:
- sortOrder:
- entryPoint:
- massupdate:
- usertheme:
- noThemeSave:
- usercolor:
- userfont:
- userthemegrouptabs:
- MSID:
호출 URL에 따른 처리 프로그램
- /sugar/include/MVC/Controller/SugarController.php#process();
- /sugar/include/MVC/View/SugarView.php#process();
- /sugar/modules/Accounts/views/view.view_명.php
- /sugar/include/MVC/View/views/view.view_명.php
/sugar/modules/Accounts/ 폴더 구조
- views/: Default. /sugar/include/MVC/View/views/
- tpls/: Default. /sugar/include/MVC/View/tpls/
- metadata/: 메타데이터
- Dashlets/: 대쉬렛
- SugarFeeds/: 피드
- language/: 언어 파일, ko_KR.lang.php
디렉토리 구조
프로세스 구조
index.php를 분석하여 SugarCRM의 프로세스 흐름을 파악한다.
require_once('include/entryPoint.php')
- require_once('config.php') : 환경 설정
- require_once('config_override.php') : 환경 설정 개인화
- require_once 'include/SugarObjects/SugarConfig.php' : SugarCRM 환경 설정
- require_once('include/utils.php')
- require_once('sugar_version.php');
- require_once('include/database/PearDatabase.php');
- require_once('include/database/DBManager.php');
- require_once('include/database/DBManagerFactory.php');
- require_once('include/dir_inc.php');
- require_once('include/Localization/Localization.php');
- require_once('include/javascript/jsAlerts.php');
- require_once('include/TimeDate.php');
- require_once('include/modules.php');
- require_once('data/SugarBean.php');
- require_once('include/utils/file_utils.php');
- require_once('include/utils/mvc_utils.php');
- require_once('include/SugarEmailAddress/SugarEmailAddress.php');
- require_once('include/SugarLogger/LoggerManager.php');
- require_once('modules/Trackers/BreadCrumbStack.php');
- require_once('modules/Trackers/Tracker.php');
- require_once('modules/Trackers/TrackerManager.php');
- require_once('modules/ACL/ACLController.php');
- require_once('modules/Administration/Administration.php');
- require_once('modules/Administration/updater_utils.php');
- require_once('modules/Users/User.php');
- require_once('modules/Users/authentication/AuthenticationController.php');
- require_once('include/utils/LogicHook.php');
- require SUGAR_PATH . '/include/SugarObjects/SugarRegistry.php';
require_once('include/MVC/SugarApplication.php')
require_once('include/MVC/Controller/ControllerFactory.php')
require_once('include/MVC/Controller/SugarController.php') : 디폴터 Controller
- require_once('include/MVC/View/SugarView.php');
- require_once('modules/Administration/updater_utils.php');
- $this->process();
- $this->processView();
:*$view->process();
require_once('custom/modules/'.$module.'/controller.php') : 모듈별 사용자 정의 Controller
require_once('modules/'.$module.'/controller.php') : Module별 Controller
$controller = new SugarController() : 디폴터 Controller
require_once('include/MVC/View/ViewFactory.php')
- require_once('include/MVC/View/SugarView.php');
- 'custom/modules/'.$module.'/views/view.'.$type.'.php : 사용자 정의 모듈별 View
- modules/'.$module.'/views/view.'.$type.'.php' : Module별 View
- custom/include/MVC/View/views/view.'.$type.'.php : 사용자 정의 디폴트 View
- include/MVC/View/views/view.'.$type.'.php : 디폴트 View
require_once('modules/Users/authentication/AuthenticationController.php')
$app->execute()
- Request parameter : module, action, usertheme, MSID, entryPoint, massupdate
- $this->controller = ControllerFactory::getController($module);
- $this->controller->checkEntryPointRequiresAuth($_REQUEST'entryPoint')
- $this->loadUser();
- $this->ACLFilter();
- $this->preProcess();
- $this->controller->preProcess();
- $this->loadLanguages();
- $this->checkDatabaseVersion();
- $this->loadDisplaySettings();
- $this->loadLicense();
- $this->loadGlobals();
- $this->setupResourceManagement($module);
- $this->controller->execute();
- sugar_cleanup();
require_once('include/resource/ResourceManager.php');
require_once ('modules/Sync/file_config.php');
require_once('modules/Sync/SyncController.php');
require_once('themes/'.$GLOBALS'theme'.'/layout_utils.php');
화면 구조
include/MVC/View/SugarView.php
- $this->displayHeader();
- Theme : header.tpl
- $this->displayHeader();
_leftFormHiddenLastViewed.tpl
_leftFormHiddenShortcuts.tpl
_companyLogo.tpl
_colorFontPicker.tpl
_globalLinks.tpl
_welcome.tpl
_headerSearch.tpl
_headerModuleListGroupTabs.tpl
_headerModuleList.tpl
_headerLastViewed.tpl
_headerShortcuts.tpl
_leftFormHide.tpl$this->preDisplay();
$this->displayErrors();
$this->display();
- Module과 Action에 해당하는 template이 사용됨
$this->displaySubPanels();
- $subpanel->display();
$this->displayFooter();
- Theme : footer.tpl
Default Theme
- Sugar <- default
Shortcut Menu
SugarCRM의 좌측 메뉴에 보여지는 Shortcut Menu (바로가기 메뉴)를 정리 한다.
/modules/모듈명/Menu.php 에 권한에 따른 바로 가기 메뉴가 정의 되어 있다.
동적 구조
- Relate 변수
select name, type, ext2, ext3
from fields_meta_data
where custom_module = 'cases'
and type = 'relate';
select custom_module, name, type, len, required, ext1, ext2, ext3, ext4
from fields_meta_data
where custom_module = 'Campaigns'
order by name;
- relationship
desc relationships;
문제 해결
- Sugar 화면이 깨어지고 Ajax 관련 오류 창이 표시됨
CentOS에 설치된 PHP 라이브러리의 버전이 달라 문제가 발생 합니다.
- jssource/Minifier.php 파일 수정
//--- 158 line, pnuskgh
// $this->input = preg_replace('/\h/u', ' ', $this->input);
$this->input = preg_replace('/[\t]()/u', ' ', $this->input);
- Admin -> Repair -> Repair JS Files 실행
- cache/ 폴더의 파일을 모두 삭제
Home 화면에서 Dashlet의 탭이 계속 실행중으로 표시됨
- cache/xml/ 폴더를 apache:apache 권한으로 생성을 합니다.
Localization
한글팩 작성
Localization 범위
- 한글 번역, 이미지 등 한글화
- 어순 : 성 이름
- 날짜, 시간
- 다중 통화 (Currency)
- 주소
PHP 파일 번역
- ko_KR.lang.php
- ~Dashlet.ko_KR.lang.php
- demoData.ko_KR.php
- phpmailer.lang-ko.php
Template 파일 번역
- LotusLiveSignup.ko_KR.tpl
- footer.tpl
JavaScript 파일 번역
- ko.js, ko_dlg.js
- calendar.js, calendar-min.js
HTML 파일 번역
- ko_KR.notify_template.html
- ko_KR.Portal.html
이미지 파일을 gimp를 사용하여 새로 작성
PDF 파일에서 한글이 깨어짐
- TCPDF에서 hysmyeongjostdmedium.php 파일을 다운로드
- include/tcpdf/fonts/ 폴더에 복사
- "Admin -> Repair -> Quick Repair and Rebuild" 메뉴 실행
- 개인별 Profile에서 PDF Settings을 다음과 같이 설정
- Font for Header and Body : MyungJo Medium (Korean)
- Font for Footer : MyungJo Medium (Korean)
- http://www.xml-convert.com/en/convert-tff-font-to-afm-pfa-fpdf-tcpdf
vi include/Sugarpdf/sugarpdf_default.php
- helvetica를 MyungJo Medium (Korean)로 변경 합니다. (미확인 사항)
'PDF_FONT_NAME_MAIN'=>'helvetica',
'PDF_FONT_NAME_DATA'=>'helvetica',
TCPDF에 ttf 폰트 추가 (실패)
- 나눔글꼴을 다운로드 합니다.
- http://www.xml-convert.com/en/convert-tff-font-to-afm-pfa-fpdf-tcpdf 사이트에서 ttf2ufm.exe 파일을 다운로드 합니다.
- ttf2ufm.exe ~.ttf 명령을 사용하여 ~.ufm 파일을 생성 합니다.
- /include/tcpdf/fonts/utils/makefont.php 파일을 수정하여 nanumgothiccoding.php, nanumgothiccoding.z, nanumgothiccoding.ctg.z 파일을 생성 합니다.
- php -q makefont.php NanumGothicCoding.ttf NanumGothicCoding.ufm
- 생성된 파일을 /include/tcpdf/fonts/ 폴더에 복사 합니다.
주소
Address Localization
- include/SugarFields/Fields/Address/ko_KR.DetailView.tpl, ko_KR.EditView.tpl
한글로 모듈(거래처) 목록 검색
- AJAX 목록 화면에서 한글로 검색시 화면이 표시되지 않음
include/ListView/ListViewData.php 파일의 529 라인에서
$queryString = htmlentities($_REQUEST[$field_name]($field_name.md));을 아래와 같이 변경함
$queryString = htmlentities($_REQUEST[$field_name]($field_name.md), null, "UTF-8");
메일 발송시 보낸 사람 이름
- 메일 발송시 보낸 사람 이름이 깨어짐
modules/Emails/Email.php 851 line
From : $this->from_addr_name = $this->from_addr;
To : $this->from_addr_name = "{$mail->FromName} <{$mail->From}>";
- 근본적으로 해결을 하려면 $mail->FromName 값이 MIME으로 인코딩 되었을 경우에만 decode를 하도록 수정할 것
Tips
Conditional Formatting on Cases List view and Dashlets, 2012.10.15
Leveraging ACL access levels in your code
참고 문헌
[Sugar Customize](Sugar Customize.md)
기타 오픈소스 CRM
지원 업체
{{지원업체}}
[[Category:오픈소스|Category:오픈소스]]
[[Category:CRM|Category:CRM]]
[[Category:한글화|Category:한글화]]
[[Category:Sugar|Category:Sugar]]
분류: WebSite
최종 수정일: 2022-10-24 19:17:28
이전글 :
다음글 :